Blame


1 00713762 2021-08-17 op ;;; vc-got.el --- VC backend for Game of Trees VCS -*- lexical-binding: t; -*-
2 af5ef7cd 2020-11-29 op
3 f0d7c7e7 2022-05-28 op ;; Copyright (C) 2021-2022 Free Software Foundation, Inc.
4 af5ef7cd 2020-11-29 op
5 76d978fa 2021-01-05 op ;; Author: Omar Polo <op@omarpolo.com>
6 4b3c0c67 2021-08-17 op ;; Timo Myyrä <timo.myyra@bittivirhe.fi>
7 78607b0c 2022-05-28 op ;; URL: https://projects.omarpolo.com/vc-got.html
8 61b9a83b 2021-08-17 op ;; Keywords: vc tools
9 18cc42ab 2022-05-28 op ;; Version: 1.1.1
10 3efc7598 2021-09-09 op ;; Package-Requires: ((emacs "25.1"))
11 af5ef7cd 2020-11-29 op
12 406d7b4d 2020-12-26 op ;; This program is free software; you can redistribute it and/or modify
13 406d7b4d 2020-12-26 op ;; it under the terms of the GNU General Public License as published by
14 406d7b4d 2020-12-26 op ;; the Free Software Foundation, either version 3 of the License, or
15 406d7b4d 2020-12-26 op ;; (at your option) any later version.
16 af5ef7cd 2020-11-29 op
17 406d7b4d 2020-12-26 op ;; This program is distributed in the hope that it will be useful,
18 406d7b4d 2020-12-26 op ;; but WITHOUT ANY WARRANTY; without even the implied warranty of
19 406d7b4d 2020-12-26 op ;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
20 406d7b4d 2020-12-26 op ;; GNU General Public License for more details.
21 af5ef7cd 2020-11-29 op
22 406d7b4d 2020-12-26 op ;; You should have received a copy of the GNU General Public License
23 406d7b4d 2020-12-26 op ;; along with this program. If not, see <https://www.gnu.org/licenses/>.
24 406d7b4d 2020-12-26 op
25 7726b6c9 2021-08-10 op ;;; Commentary:
26 af5ef7cd 2020-11-29 op
27 b3b5f78a 2022-05-28 op ;; vc-got is a VC backend for the Game of Trees (got) version control
28 b3b5f78a 2022-05-28 op ;; system.
29 b3b5f78a 2022-05-28 op ;;
30 af5ef7cd 2020-11-29 op ;; Backend implementation status
31 af5ef7cd 2020-11-29 op ;;
32 af5ef7cd 2020-11-29 op ;; Function marked with `*' are required, those with `-' are optional.
33 af5ef7cd 2020-11-29 op ;;
34 af5ef7cd 2020-11-29 op ;; FUNCTION NAME STATUS
35 af5ef7cd 2020-11-29 op ;;
36 af5ef7cd 2020-11-29 op ;; BACKEND PROPERTIES:
37 af5ef7cd 2020-11-29 op ;; * revision-granularity DONE
38 af5ef7cd 2020-11-29 op ;; - update-on-retrieve-tag XXX: what should this do?
39 af5ef7cd 2020-11-29 op ;;
40 af5ef7cd 2020-11-29 op ;; STATE-QUERYING FUNCTIONS:
41 af5ef7cd 2020-11-29 op ;; * registered DONE
42 af5ef7cd 2020-11-29 op ;; * state DONE
43 af5ef7cd 2020-11-29 op ;; - dir-status-files DONE
44 a8466f02 2020-12-07 op ;; - dir-extra-headers DONE
45 2d83de2e 2021-01-05 op ;; - dir-printer DONE
46 af5ef7cd 2020-11-29 op ;; - status-fileinfo-extra NOT IMPLEMENTED
47 af5ef7cd 2020-11-29 op ;; * working-revision DONE
48 af5ef7cd 2020-11-29 op ;; * checkout-model DONE
49 694534b4 2020-12-05 op ;; - mode-line-string DONE
50 23a0b465 2020-11-30 op ;;
51 23a0b465 2020-11-30 op ;; STATE-CHANGING FUNCTIONS:
52 23a0b465 2020-11-30 op ;; * create-repo NOT IMPLEMENTED
53 eb85ad27 2020-12-05 op ;; I don't think got init does what this function is supposed to
54 eb85ad27 2020-12-05 op ;; do.
55 23a0b465 2020-11-30 op ;; * register DONE
56 23a0b465 2020-11-30 op ;; - responsible-p DONE
57 3ec8a8b4 2021-08-17 op ;; - receive-file NOT NEEDED, default `register' is fine
58 241a3b12 2021-08-11 op ;; - unregister DONE
59 23a0b465 2020-11-30 op ;; * checkin DONE
60 23a0b465 2020-11-30 op ;; * find-revision DONE
61 eb85ad27 2020-12-05 op ;; * checkout NOT IMPLEMENTED
62 eb85ad27 2020-12-05 op ;; I'm not sure how to properly implement this. Does filling
63 eb85ad27 2020-12-05 op ;; FILE with the find-revision do the trick? Or use got update?
64 eb85ad27 2020-12-05 op ;; * revert DONE
65 eb85ad27 2020-12-05 op ;; - merge-file NOT IMPLEMENTED
66 eb85ad27 2020-12-05 op ;; - merge-branch DONE
67 eb85ad27 2020-12-05 op ;; - merge-news NOT IMPLEMENTED
68 eb85ad27 2020-12-05 op ;; - pull DONE
69 686eac9a 2020-12-07 op ;; - push DONE
70 3cdb0759 2020-01-03 47739920+ ;; - steal-lock NOT NEEDED, `got' is not using locks
71 eb85ad27 2020-12-05 op ;; - modify-change-comment NOT IMPLEMENTED
72 eb85ad27 2020-12-05 op ;; can be implemented via histedit, if I understood correctly
73 eb85ad27 2020-12-05 op ;; what it is supposed to do.
74 ae2ad572 2021-08-17 op ;; - mark-resolved NOT NEEDED
75 ae2ad572 2021-08-17 op ;; got notice by itself when a file doesn't have any pending
76 ae2ad572 2021-08-17 op ;; conflicts to be resolved.
77 ae2ad572 2021-08-17 op ;; - find-admin-dir NOT NEEDED
78 eb85ad27 2020-12-05 op ;;
79 eb85ad27 2020-12-05 op ;; HISTORY FUNCTIONS
80 eb85ad27 2020-12-05 op ;; * print-log DONE
81 9ad1eb5d 2020-12-07 op ;; * log-outgoing DONE
82 6eda2f1f 2020-12-26 op ;; * log-incoming DONE
83 eb85ad27 2020-12-05 op ;; - log-search DONE
84 0a3316a6 2021-02-22 op ;; - log-view-mode DONE
85 6eda2f1f 2020-12-26 op ;; - show-log-entry NOT IMPLEMENTED
86 6eda2f1f 2020-12-26 op ;; - comment-history NOT IMPLEMENTED
87 6eda2f1f 2020-12-26 op ;; - update-changelog NOT IMPLEMENTED
88 7c257a7b 2020-12-31 noreply ;; * diff DONE
89 5fb2f474 2021-08-10 op ;; - revision-completion-table DONE
90 3f25d9ff 2020-01-02 noreply ;; - annotate-command DONE
91 3f25d9ff 2020-01-02 noreply ;; - annotate-time DONE
92 424f5152 2021-08-10 op ;; - annotate-current-time NOT NEEDED
93 424f5152 2021-08-10 op ;; the default time handling is enough.
94 3f25d9ff 2020-01-02 noreply ;; - annotate-extract-revision-at-line DONE
95 7c257a7b 2020-12-31 noreply ;; - region-history NOT IMPLEMENTED
96 7c257a7b 2020-12-31 noreply ;; - region-history-mode NOT IMPLEMENTED
97 7c257a7b 2020-12-31 noreply ;; - mergebase NOT IMPLEMENTED
98 7c257a7b 2020-12-31 noreply ;;
99 7c257a7b 2020-12-31 noreply ;; TAG SYSTEM
100 6ab1c7b4 2021-08-11 op ;; - create-tag DONE
101 09d59379 2021-08-11 op ;; - retrieve-tag DONE
102 7c257a7b 2020-12-31 noreply ;;
103 7c257a7b 2020-12-31 noreply ;; MISCELLANEOUS NOT IMPLEMENTED
104 3cdb0759 2020-01-03 47739920+ ;; - make-version-backups-p NOT NEEDED, `got' works fine locally
105 7c257a7b 2020-12-31 noreply ;; - root DONE
106 49945a03 2021-08-11 op ;; - ignore NOT NEEDED, the default action is good
107 49945a03 2021-08-11 op ;; - ignore-completion-table NOT NEEDED, the default action is good
108 2552a6c7 2021-08-11 op ;; - find-ignore-file DONE
109 3f25d9ff 2020-01-02 noreply ;; - previous-revision DONE
110 3f25d9ff 2020-01-02 noreply ;; - next-revision DONE
111 7c257a7b 2020-12-31 noreply ;; - log-edit-mode NOT IMPLEMENTED
112 3cdb0759 2020-01-03 47739920+ ;; - check-headers NOT NEEDED, `got' does not use headers
113 a96df0e8 2020-01-03 noreply ;; - delete-file DONE
114 3ec8a8b4 2021-08-17 op ;; - rename-file NOT IMPLEMENTED
115 c151c257 2021-08-03 op ;; - find-file-hook DONE
116 3ec8a8b4 2021-08-17 op ;; - extra-menu NOT IMPLEMENTED
117 3cdb0759 2020-01-03 47739920+ ;; - extra-dir-menu NOT IMPLEMENTED, same as above
118 7bb16f79 2020-01-03 op ;; - conflicted-files DONE
119 8b635a9f 2020-01-03 op ;; - repository-url DONE
120 b3b5f78a 2022-05-28 op
121 b3b5f78a 2022-05-28 op ;;; Code:
122 af5ef7cd 2020-11-29 op
123 eb85ad27 2020-12-05 op ;; TODO: vc-git has most function that starts with:
124 eb85ad27 2020-12-05 op ;;
125 eb85ad27 2020-12-05 op ;; (let* ((root (vc-git-root default-directory))
126 eb85ad27 2020-12-05 op ;; (buffer (format "*vc-git : %s*" (expand-file-name root)))
127 eb85ad27 2020-12-05 op ;; ...)
128 eb85ad27 2020-12-05 op ;; ...)
129 eb85ad27 2020-12-05 op ;;
130 eb85ad27 2020-12-05 op ;; we should 1) investigate if also other backends do something like
131 eb85ad27 2020-12-05 op ;; this (or if there is a better way) and 2) try to do the same.
132 af5ef7cd 2020-11-29 op
133 af5ef7cd 2020-11-29 op (eval-when-compile
134 af5ef7cd 2020-11-29 op (require 'subr-x))
135 af5ef7cd 2020-11-29 op
136 af5ef7cd 2020-11-29 op (require 'cl-lib)
137 af5ef7cd 2020-11-29 op (require 'seq)
138 23a0b465 2020-11-30 op (require 'vc)
139 3f25d9ff 2020-01-02 noreply (require 'vc-annotate)
140 f7012697 2021-08-17 op
141 f7012697 2021-08-17 op ;; FIXME: avoid loading this? We only need it for
142 f7012697 2021-08-17 op ;; log-edit-extract-headers in vc-got-checkin.
143 f7012697 2021-08-17 op (require 'log-edit)
144 6b6c8a78 2021-01-04 op
145 2d83de2e 2021-01-05 op ;; FIXME: avoid loading this? We only need it for
146 2d83de2e 2021-01-05 op ;; vc-dir-filename-mouse-map in our custom printer.
147 2d83de2e 2021-01-05 op (require 'vc-dir)
148 ba123905 2020-12-08 op
149 95a78ecb 2021-08-31 op ;; FIXME: avoid loading this? We only need it for
150 95a78ecb 2021-08-31 op ;; compilation-{directory,arguments}.
151 95a78ecb 2021-08-31 op (require 'compile)
152 95a78ecb 2021-08-31 op
153 95a78ecb 2021-08-31 op ;; FIXME: avoid loading this? We only need it for
154 95a78ecb 2021-08-31 op ;; log-view-{file-re,per-file-logs,message-re}.
155 95a78ecb 2021-08-31 op (require 'log-view)
156 95a78ecb 2021-08-31 op
157 e79563bd 2020-12-31 noreply (defgroup vc-got nil
158 e79563bd 2020-12-31 noreply "VC GoT backend."
159 e79563bd 2020-12-31 noreply :group 'vc)
160 af5ef7cd 2020-11-29 op
161 e79563bd 2020-12-31 noreply (defcustom vc-got-program "got"
162 e79563bd 2020-12-31 noreply "Name of the Got executable (excluding any arguments)."
163 6eea6cfb 2021-01-05 op :type 'string)
164 e79563bd 2020-12-31 noreply
165 12ca62f2 2020-12-26 op (defcustom vc-got-diff-switches t
166 12ca62f2 2020-12-26 op "String or list of strings specifying switches for Got diff under VC.
167 12ca62f2 2020-12-26 op If nil, use the value of `vc-diff-switches'. If t, use no switches."
168 12ca62f2 2020-12-26 op :type '(choice (const :tag "Unspecified" nil)
169 12ca62f2 2020-12-26 op (const :tag "None" t)
170 12ca62f2 2020-12-26 op (string :tag "Argument String")
171 6eea6cfb 2021-01-05 op (repeat :tag "Argument List" :value ("") string)))
172 12ca62f2 2020-12-26 op
173 af5ef7cd 2020-11-29 op ;; helpers
174 95a78ecb 2021-08-31 op (defmacro vc-got--with-emacs-version<= (version &rest body)
175 95a78ecb 2021-08-31 op "Eval BODY only when the Emacs version in greater or equal VERSION."
176 95a78ecb 2021-08-31 op (declare (debug body)
177 95a78ecb 2021-08-31 op (indent defun))
178 95a78ecb 2021-08-31 op (when (version<= version emacs-version)
179 95a78ecb 2021-08-31 op `(progn ,@body)))
180 95a78ecb 2021-08-31 op
181 f09d6359 2020-12-31 noreply (defun vc-got--program-version ()
182 76d978fa 2021-01-05 op "Return string representing the got version."
183 f09d6359 2020-12-31 noreply (let (process-file-side-effects)
184 f09d6359 2020-12-31 noreply (with-temp-buffer
185 f09d6359 2020-12-31 noreply (vc-got--call "-V")
186 f09d6359 2020-12-31 noreply (substring (buffer-string) 4 -1))))
187 af5ef7cd 2020-11-29 op
188 af5ef7cd 2020-11-29 op (defun vc-got-root (file)
189 af5ef7cd 2020-11-29 op "Return the work tree root for FILE, or nil."
190 6a973a79 2021-08-11 op (vc-find-root file ".got"))
191 af5ef7cd 2020-11-29 op
192 af5ef7cd 2020-11-29 op (defmacro vc-got-with-worktree (file &rest body)
193 af5ef7cd 2020-11-29 op "Evaluate BODY in the work tree directory of FILE."
194 3efc7598 2021-09-09 op (declare (debug t) (indent defun))
195 af5ef7cd 2020-11-29 op `(when-let (default-directory (vc-got-root ,file))
196 af5ef7cd 2020-11-29 op ,@body))
197 686eac9a 2020-12-07 op
198 686eac9a 2020-12-07 op (defun vc-got--repo-root ()
199 686eac9a 2020-12-07 op "Return the path to the repository root.
200 686eac9a 2020-12-07 op Assume `default-directory' is inside a got worktree."
201 686eac9a 2020-12-07 op (vc-got-with-worktree default-directory
202 686eac9a 2020-12-07 op (with-temp-buffer
203 686eac9a 2020-12-07 op (insert-file-contents ".got/repository")
204 76d978fa 2021-01-05 op (string-trim (buffer-string) "" "\n"))))
205 af5ef7cd 2020-11-29 op
206 af5ef7cd 2020-11-29 op (defun vc-got--call (&rest args)
207 76d978fa 2021-01-05 op "Call `vc-got-program' with ARGS.
208 76d978fa 2021-01-05 op The output will be placed in the current buffer."
209 fab791a2 2020-01-03 op (apply #'process-file vc-got-program nil (current-buffer) nil
210 3efc7598 2021-09-09 op (apply #'nconc (mapcar (lambda (s) (if (listp s) s (list s))) args))))
211 af5ef7cd 2020-11-29 op
212 23a0b465 2020-11-30 op (defun vc-got--add (files)
213 23a0b465 2020-11-30 op "Add FILES to got, passing `vc-register-switches' to the command invocation."
214 23a0b465 2020-11-30 op (with-temp-buffer
215 c3185bf7 2021-08-11 op (vc-got--call "add" vc-register-switches "--" files)))
216 23a0b465 2020-11-30 op
217 0a66694e 2020-01-02 op (defun vc-got--log (&optional path limit start-commit stop-commit
218 32819756 2021-09-14 op search-pattern reverse include-diff)
219 76d978fa 2021-01-05 op "Execute the log command in the worktree of PATH in the current buffer.
220 23a0b465 2020-11-30 op LIMIT limits the maximum number of commit returned.
221 23a0b465 2020-11-30 op
222 518ede14 2020-12-05 op START-COMMIT: start traversing history at the specified commit.
223 9ad1eb5d 2020-12-07 op STOP-COMMIT: stop traversing history at the specified commit.
224 518ede14 2020-12-05 op SEARCH-PATTERN: limit to log messages matched by the regexp given.
225 0a66694e 2020-01-02 op REVERSE: display the log messages in reverse order.
226 32819756 2021-09-14 op INCLUDE-DIFF: display the patch of modifications made in each commit.
227 518ede14 2020-12-05 op
228 23a0b465 2020-11-30 op Return nil if the command failed or if PATH isn't included in any
229 23a0b465 2020-11-30 op worktree."
230 3efc7598 2021-09-09 op (let ((process-file-side-effects nil))
231 0a66694e 2020-01-02 op (vc-got-with-worktree (or path default-directory)
232 0a3316a6 2021-02-22 op (when (zerop
233 0a3316a6 2021-02-22 op (save-excursion
234 0a3316a6 2021-02-22 op (vc-got--call "log"
235 3efc7598 2021-09-09 op (and limit (list "-l" (format "%s" limit)))
236 3efc7598 2021-09-09 op (and start-commit (list "-c" start-commit))
237 3efc7598 2021-09-09 op (and stop-commit (list "-x" stop-commit))
238 3efc7598 2021-09-09 op (and search-pattern (list "-s" search-pattern))
239 3efc7598 2021-09-09 op (and reverse '("-R"))
240 32819756 2021-09-14 op (and include-diff '("-p"))
241 32819756 2021-09-14 op ;; "--"
242 0a3316a6 2021-02-22 op path)))
243 0a3316a6 2021-02-22 op (save-excursion
244 3ec8a8b4 2021-08-17 op (delete-matching-lines
245 3ec8a8b4 2021-08-17 op "^-----------------------------------------------$")
246 0a3316a6 2021-02-22 op t)))))
247 af5ef7cd 2020-11-29 op
248 887e0699 2021-02-22 op (defun vc-got--status (status-codes dir-or-file &optional files)
249 dd4e6a2c 2022-05-28 op "Return a list of lists (FILE STATUS STAGE-STATUS).
250 af5ef7cd 2020-11-29 op DIR-OR-FILE can be either a directory or a file. If FILES is
251 af5ef7cd 2020-11-29 op given, return the status of those files, otherwise the status of
252 668dc8eb 2021-01-04 op DIR-OR-FILE. STATUS-CODES is either nil, or a string that's
253 668dc8eb 2021-01-04 op passed as the -s flag to got status to limit the types of status
254 668dc8eb 2021-01-04 op to report (e.g. \"CD\" to report only conflicts and deleted
255 668dc8eb 2021-01-04 op files)."
256 2d039a22 2021-02-21 op (with-temp-buffer
257 2d039a22 2021-02-21 op (let* ((default-directory (expand-file-name
258 2d039a22 2021-02-21 op (if (file-directory-p dir-or-file)
259 2d039a22 2021-02-21 op dir-or-file
260 2d039a22 2021-02-21 op (file-name-directory dir-or-file))))
261 2d039a22 2021-02-21 op (root (vc-got-root default-directory))
262 2d039a22 2021-02-21 op (process-file-side-effects))
263 2d039a22 2021-02-21 op (when (zerop (vc-got--call "status"
264 3efc7598 2021-09-09 op (and status-codes (list "-s" status-codes))
265 c3185bf7 2021-08-11 op "--"
266 2d039a22 2021-02-21 op (or files dir-or-file)))
267 2d039a22 2021-02-21 op (goto-char (point-min))
268 2d039a22 2021-02-21 op (cl-loop until (eobp)
269 3ec8a8b4 2021-08-17 op collect (vc-got--parse-status-line root)
270 2d039a22 2021-02-21 op do (forward-line))))))
271 3ec8a8b4 2021-08-17 op
272 3ec8a8b4 2021-08-17 op (defun vc-got--parse-status-line (root)
273 3ec8a8b4 2021-08-17 op "Parse a line of the the output of status.
274 3ec8a8b4 2021-08-17 op ROOT is the root of the repo."
275 3ec8a8b4 2021-08-17 op ;; the format of each line is
276 3ec8a8b4 2021-08-17 op ;; <status-char> <stage-char> <spc> <filename> \n
277 3ec8a8b4 2021-08-17 op (let* ((file-status (prog1 (vc-got--parse-status-char
278 3ec8a8b4 2021-08-17 op (char-after))
279 3ec8a8b4 2021-08-17 op (forward-char)))
280 3ec8a8b4 2021-08-17 op (stage-status (let* ((c (char-after)))
281 3ec8a8b4 2021-08-17 op (prog1
282 3ec8a8b4 2021-08-17 op (when (member c '(?M ?A ?D))
283 3ec8a8b4 2021-08-17 op c)
284 3ec8a8b4 2021-08-17 op (forward-char))))
285 3ec8a8b4 2021-08-17 op (filename (progn
286 3ec8a8b4 2021-08-17 op (forward-char)
287 3ec8a8b4 2021-08-17 op (buffer-substring (point)
288 3ec8a8b4 2021-08-17 op (line-end-position)))))
289 3ec8a8b4 2021-08-17 op (list (file-relative-name (expand-file-name filename root)
290 3ec8a8b4 2021-08-17 op default-directory)
291 3ec8a8b4 2021-08-17 op (or file-status (and stage-status 'up-to-date))
292 3ec8a8b4 2021-08-17 op stage-status)))
293 af5ef7cd 2020-11-29 op
294 668dc8eb 2021-01-04 op (defun vc-got--parse-status-char (c)
295 668dc8eb 2021-01-04 op "Parse status char C into a symbol accepted by `vc-state'."
296 668dc8eb 2021-01-04 op (cl-case c
297 668dc8eb 2021-01-04 op (?M 'edited)
298 668dc8eb 2021-01-04 op (?A 'added)
299 668dc8eb 2021-01-04 op (?D 'removed)
300 668dc8eb 2021-01-04 op (?C 'conflict)
301 668dc8eb 2021-01-04 op (?! 'missing)
302 76d978fa 2021-01-05 op (?~ 'edited) ; XXX: what does it means for a file to be ``obstructed''?
303 668dc8eb 2021-01-04 op (?? 'unregistered)
304 76d978fa 2021-01-05 op (?m 'edited) ; modified file modes
305 668dc8eb 2021-01-04 op (?N nil)))
306 23a0b465 2020-11-30 op
307 23a0b465 2020-11-30 op (defun vc-got--cat (commit obj-id)
308 23a0b465 2020-11-30 op "Execute got cat -c COMMIT OBJ-ID in the current buffer."
309 76d978fa 2021-01-05 op (let (process-file-side-effects)
310 76d978fa 2021-01-05 op (zerop (vc-got--call "cat" "-c" commit obj-id))))
311 eb85ad27 2020-12-05 op
312 eb85ad27 2020-12-05 op (defun vc-got--revert (&rest files)
313 76d978fa 2021-01-05 op "Execute got revert FILES."
314 eb85ad27 2020-12-05 op (vc-got-with-worktree (car files)
315 eb85ad27 2020-12-05 op (with-temp-buffer
316 c3185bf7 2021-08-11 op (zerop (vc-got--call "revert" "--" files)))))
317 eb85ad27 2020-12-05 op
318 eb85ad27 2020-12-05 op (defun vc-got--list-branches ()
319 eb85ad27 2020-12-05 op "Return an alist of (branch . commit)."
320 76d978fa 2021-01-05 op (let (process-file-side-effects)
321 76d978fa 2021-01-05 op (with-temp-buffer
322 76d978fa 2021-01-05 op (when (zerop (vc-got--call "branch" "-l"))
323 3efc7598 2021-09-09 op (let (alist)
324 3efc7598 2021-09-09 op (goto-char (point-min))
325 660cf35d 2021-09-25 op (while (re-search-forward "^\\*?[[:space:]]+\\(.+\\): \\([[:word:]]+\\)$"
326 660cf35d 2021-09-25 op nil t)
327 3efc7598 2021-09-09 op (push (cons (match-string 1) (match-string 2)) alist))
328 3efc7598 2021-09-09 op alist)))))
329 eb85ad27 2020-12-05 op
330 694534b4 2020-12-05 op (defun vc-got--current-branch ()
331 694534b4 2020-12-05 op "Return the current branch."
332 76d978fa 2021-01-05 op (let (process-file-side-effects)
333 76d978fa 2021-01-05 op (with-temp-buffer
334 76d978fa 2021-01-05 op (when (zerop (vc-got--call "branch"))
335 76d978fa 2021-01-05 op (string-trim (buffer-string) "" "\n")))))
336 eb85ad27 2020-12-05 op
337 eb85ad27 2020-12-05 op (defun vc-got--integrate (branch)
338 eb85ad27 2020-12-05 op "Integrate BRANCH into the current one."
339 eb85ad27 2020-12-05 op (with-temp-buffer
340 76d978fa 2021-01-05 op (zerop (vc-got--call "integrate" branch))))
341 db89d6fd 2021-08-16 op
342 db89d6fd 2021-08-16 op (defun vc-got--update (branch &optional paths)
343 db89d6fd 2021-08-16 op "Update to a different commit or BRANCH.
344 db89d6fd 2021-08-16 op Optionally restrict the update operation to files at or within
345 db89d6fd 2021-08-16 op the specified PATHS."
346 db89d6fd 2021-08-16 op (with-temp-buffer
347 db89d6fd 2021-08-16 op (unless (zerop (vc-got--call "update" "-b" branch "--" paths))
348 db89d6fd 2021-08-16 op (error "[vc-got] can't update to branch %s: %s"
349 db89d6fd 2021-08-16 op branch
350 db89d6fd 2021-08-16 op (buffer-string)))))
351 23a0b465 2020-11-30 op
352 45f7f941 2021-11-09 op (defun vc-got--diff-files (files)
353 36e5655f 2021-11-09 op "Compute the local modifications to FILES."
354 76d978fa 2021-01-05 op (let (process-file-side-effects)
355 36e5655f 2021-11-09 op (zerop (vc-got--call "diff" (vc-switches 'got 'diff) "-P" "--"
356 36e5655f 2021-11-09 op files))))
357 6b6c8a78 2021-01-04 op
358 36e5655f 2021-11-09 op (defun vc-got--diff-objects (obj1 obj2)
359 36e5655f 2021-11-09 op "Diff the two objects OBJ1 and OBJ2.
360 36e5655f 2021-11-09 op OBJ1 and OBJ2 are interpreted as a reference, tag name, or an
361 36e5655f 2021-11-09 op object ID SHA1 hash."
362 36e5655f 2021-11-09 op (let (process-file-side-effects)
363 36e5655f 2021-11-09 op (zerop (vc-got--call "diff" (vc-switches 'got 'diff) "--" obj1 obj2))))
364 36e5655f 2021-11-09 op
365 6b6c8a78 2021-01-04 op (defun vc-got--unstage (file-or-directory)
366 6b6c8a78 2021-01-04 op "Unstage all the staged hunks at or within FILE-OR-DIRECTORY.
367 6b6c8a78 2021-01-04 op If it's nil, unstage every staged changes across the entire work
368 6b6c8a78 2021-01-04 op tree."
369 c3185bf7 2021-08-11 op (zerop (vc-got--call "unstage" "--" file-or-directory)))
370 a96df0e8 2020-01-03 noreply
371 a96df0e8 2020-01-03 noreply (defun vc-got--remove (file &optional force keep-local)
372 76d978fa 2021-01-05 op "Use got to remove FILE.
373 76d978fa 2021-01-05 op If FORCE is non-nil perform the operation even if a file contains
374 76d978fa 2021-01-05 op local modification. If KEEP-LOCAL is non-nil keep the affected
375 76d978fa 2021-01-05 op files on disk."
376 a96df0e8 2020-01-03 noreply (vc-got-with-worktree (or file default-directory)
377 a96df0e8 2020-01-03 noreply (with-temp-buffer
378 76d978fa 2021-01-05 op (zerop (vc-got--call "remove"
379 3efc7598 2021-09-09 op (and force "-f")
380 3efc7598 2021-09-09 op (and keep-local "-k")
381 c3185bf7 2021-08-11 op "--"
382 76d978fa 2021-01-05 op file)))))
383 eb85ad27 2020-12-05 op
384 5fb2f474 2021-08-10 op (defun vc-got--ref ()
385 5fb2f474 2021-08-10 op "Return a list of all references."
386 3efc7598 2021-09-09 op (let ((process-file-side-effects nil)
387 5fb2f474 2021-08-10 op (re "^refs/\\(heads\\|remotes\\|tags\\)/\\(.*\\):")
388 5fb2f474 2021-08-10 op ;; hardcoding HEAD because it's always present and the regexp
389 5fb2f474 2021-08-10 op ;; won't match it.
390 5fb2f474 2021-08-10 op (table (list "HEAD")))
391 5fb2f474 2021-08-10 op (vc-got-with-worktree default-directory
392 5fb2f474 2021-08-10 op (with-temp-buffer
393 5fb2f474 2021-08-10 op (when (zerop (vc-got--call "ref" "-l"))
394 5fb2f474 2021-08-10 op (goto-char (point-min))
395 5fb2f474 2021-08-10 op (while (re-search-forward re nil t)
396 5fb2f474 2021-08-10 op (push (match-string 2) table))
397 5fb2f474 2021-08-10 op table)))))
398 8014bde1 2021-08-10 op
399 8014bde1 2021-08-10 op (defun vc-got--branch (name)
400 8014bde1 2021-08-10 op "Try to create and switch to the branch called NAME."
401 8014bde1 2021-08-10 op (let (process-file-side-effects)
402 8014bde1 2021-08-10 op (vc-got-with-worktree default-directory
403 8014bde1 2021-08-10 op (with-temp-buffer
404 f0d7c7e7 2022-05-28 op (or (zerop (vc-got--call "branch" "--" name))
405 f0d7c7e7 2022-05-28 op (error "[vc-got] can't create branch %s: %s" name
406 f0d7c7e7 2022-05-28 op (buffer-string)))))))
407 5fb2f474 2021-08-10 op
408 af5ef7cd 2020-11-29 op
409 af5ef7cd 2020-11-29 op ;; Backend properties
410 af5ef7cd 2020-11-29 op
411 af5ef7cd 2020-11-29 op (defun vc-got-revision-granularity ()
412 af5ef7cd 2020-11-29 op "Got has REPOSITORY granularity."
413 af5ef7cd 2020-11-29 op 'repository)
414 af5ef7cd 2020-11-29 op
415 af5ef7cd 2020-11-29 op (defun vc-got-update-on-retrieve-tag ()
416 84beb0d4 2021-08-17 op "Like vc-git, vc-got don't need to buffers on `retrieve-tag'."
417 af5ef7cd 2020-11-29 op nil)
418 af5ef7cd 2020-11-29 op
419 af5ef7cd 2020-11-29 op
420 af5ef7cd 2020-11-29 op ;; State-querying functions
421 af5ef7cd 2020-11-29 op
422 af5ef7cd 2020-11-29 op ;;;###autoload (defun vc-got-registered (file)
423 af5ef7cd 2020-11-29 op ;;;###autoload "Return non-nil if FILE is registered with got."
424 af5ef7cd 2020-11-29 op ;;;###autoload (when (vc-find-root file ".got")
425 af5ef7cd 2020-11-29 op ;;;###autoload (load "vc-got" nil t)
426 af5ef7cd 2020-11-29 op ;;;###autoload (vc-got-registered file)))
427 af5ef7cd 2020-11-29 op
428 af5ef7cd 2020-11-29 op (defun vc-got-registered (file)
429 af5ef7cd 2020-11-29 op "Return non-nil if FILE is registered with got."
430 af5ef7cd 2020-11-29 op (if (file-directory-p file)
431 76d978fa 2021-01-05 op nil ; got doesn't track directories
432 4093d2f9 2020-12-04 op (when (vc-find-root file ".got")
433 668dc8eb 2021-01-04 op (let ((s (vc-got-state file)))
434 f0d7c7e7 2022-05-28 op (not (memq s '(nil unregistered)))))))
435 af5ef7cd 2020-11-29 op
436 af5ef7cd 2020-11-29 op (defun vc-got-state (file)
437 af5ef7cd 2020-11-29 op "Return the current version control state of FILE. See `vc-state'."
438 af5ef7cd 2020-11-29 op (unless (file-directory-p file)
439 76d978fa 2021-01-05 op (let (process-file-side-effects)
440 76d978fa 2021-01-05 op ;; Manually calling got status and checking the result inline to
441 76d978fa 2021-01-05 op ;; avoid building the data structure in vc-got--status.
442 76d978fa 2021-01-05 op (with-temp-buffer
443 c3185bf7 2021-08-11 op (when (zerop (vc-got--call "status" "--" file))
444 76d978fa 2021-01-05 op (goto-char (point-min))
445 76d978fa 2021-01-05 op (if (eobp)
446 76d978fa 2021-01-05 op 'up-to-date
447 76d978fa 2021-01-05 op (vc-got--parse-status-char (char-after))))))))
448 dcb5b83a 2021-01-08 op
449 dcb5b83a 2021-01-08 op (defun vc-got--dir-filter-files (files)
450 dcb5b83a 2021-01-08 op "Remove ., .. and .got from FILES."
451 dcb5b83a 2021-01-08 op (cl-loop for file in files
452 dcb5b83a 2021-01-08 op unless (or (string= file "..")
453 dcb5b83a 2021-01-08 op (string= file ".")
454 dcb5b83a 2021-01-08 op (string= file ".got"))
455 dcb5b83a 2021-01-08 op collect file))
456 af5ef7cd 2020-11-29 op
457 af5ef7cd 2020-11-29 op (defun vc-got-dir-status-files (dir files update-function)
458 5b4a6b90 2021-01-04 op "Build the status for FILES in DIR.
459 76d978fa 2021-01-05 op The builded result is given to the callback UPDATE-FUNCTION. If
460 5b4a6b90 2021-01-04 op FILES is nil, consider all the files in DIR."
461 dcb5b83a 2021-01-08 op (let* ((fs (vc-got--dir-filter-files (or files (directory-files dir))))
462 d346ca63 2021-01-08 op ;; XXX: we call with files, wich will probably be nil on the
463 d346ca63 2021-01-08 op ;; first run, so we catch deleted, missing and edited files
464 d346ca63 2021-01-08 op ;; in subdirectories.
465 d346ca63 2021-01-08 op (res (vc-got--status nil dir files))
466 d346ca63 2021-01-08 op double-check)
467 82289421 2020-01-02 op (cl-loop for file in fs
468 76d978fa 2021-01-05 op do (when (and (not (cdr (assoc file res #'string=)))
469 76d978fa 2021-01-05 op (not (file-directory-p file))
470 76d978fa 2021-01-05 op ;; if file doesn't exists, it's a
471 76d978fa 2021-01-05 op ;; untracked file that was removed.
472 76d978fa 2021-01-05 op (file-exists-p file))
473 d346ca63 2021-01-08 op ;; if we don't know the status of a file here, it's
474 d346ca63 2021-01-08 op ;; either up-to-date or ignored. Save it for a
475 d346ca63 2021-01-08 op ;; double check
476 d346ca63 2021-01-08 op (push file double-check)))
477 cb973832 2021-01-08 op (cl-loop with statuses = (vc-got--status nil dir double-check)
478 cb973832 2021-01-08 op for file in double-check
479 cb973832 2021-01-08 op unless (eq 'unregistered (cadr (assoc file statuses #'string=)))
480 d346ca63 2021-01-08 op do (push (list file 'up-to-date nil) res))
481 d346ca63 2021-01-08 op (funcall update-function res nil)))
482 af5ef7cd 2020-11-29 op
483 c27df03e 2020-01-03 op (defun vc-got-dir-extra-headers (dir)
484 c27df03e 2020-01-03 op "Return a string for the `vc-dir' buffer heading for directory DIR."
485 76d978fa 2021-01-05 op (let ((remote (vc-got-repository-url dir)))
486 76d978fa 2021-01-05 op (concat (propertize "Repository : " 'face 'font-lock-type-face)
487 76d978fa 2021-01-05 op (vc-got--repo-root) "\n"
488 76d978fa 2021-01-05 op (when remote
489 76d978fa 2021-01-05 op (concat
490 76d978fa 2021-01-05 op (propertize "Remote URL : " 'face 'font-lock-type-face)
491 76d978fa 2021-01-05 op (vc-got-repository-url dir) "\n"))
492 76d978fa 2021-01-05 op (propertize "Branch : " 'face 'font-lock-type-face)
493 76d978fa 2021-01-05 op (vc-got--current-branch))))
494 2d83de2e 2021-01-05 op
495 2d83de2e 2021-01-05 op (defun vc-got-dir-printer (info)
496 2d83de2e 2021-01-05 op "Pretty-printer for the vc-dir-fileinfo structure INFO."
497 2d83de2e 2021-01-05 op (let* ((isdir (vc-dir-fileinfo->directory info))
498 76d978fa 2021-01-05 op (state (if isdir "" (vc-dir-fileinfo->state info)))
499 76d978fa 2021-01-05 op (stage-state (vc-dir-fileinfo->extra info))
500 76d978fa 2021-01-05 op (filename (vc-dir-fileinfo->name info)))
501 2d83de2e 2021-01-05 op (insert
502 1b2afa91 2021-08-17 op " "
503 2d83de2e 2021-01-05 op (propertize
504 2d83de2e 2021-01-05 op (format "%c" (if (vc-dir-fileinfo->marked info) ?* ? ))
505 2d83de2e 2021-01-05 op 'face 'font-lock-type-face)
506 1b2afa91 2021-08-17 op " "
507 9e019413 2021-01-05 op (propertize
508 1b2afa91 2021-08-17 op (format "%-12s" state)
509 2d83de2e 2021-01-05 op 'face (cond ((eq state 'up-to-date) 'font-lock-builtin-face)
510 76d978fa 2021-01-05 op ((memq state '(missing conflict)) 'font-lock-warning-face)
511 76d978fa 2021-01-05 op ((eq state 'edited) 'font-lock-constant-face)
512 76d978fa 2021-01-05 op (t 'font-lock-variable-name-face))
513 1b2afa91 2021-08-17 op 'mouse-face 'highlight
514 95a78ecb 2021-08-31 op 'keymap (vc-got--with-emacs-version<= "28.0.50"
515 95a78ecb 2021-08-31 op vc-dir-status-mouse-map))
516 1b2afa91 2021-08-17 op " " (propertize
517 1b2afa91 2021-08-17 op (if stage-state
518 1b2afa91 2021-08-17 op (format "%c" stage-state)
519 1b2afa91 2021-08-17 op " ")
520 1b2afa91 2021-08-17 op 'face (cond ((memq stage-state '(?A ?E)) 'font-lock-constant-face)
521 1b2afa91 2021-08-17 op ((eq stage-state ?R) 'font-lock-warning-face)
522 1b2afa91 2021-08-17 op (t 'font-lock-variable-name-face)))
523 1b2afa91 2021-08-17 op " "
524 1b2afa91 2021-08-17 op (propertize filename
525 1b2afa91 2021-08-17 op 'face (if isdir 'font-lock-comment-delimiter-face
526 1b2afa91 2021-08-17 op 'font-lock-function-name-face)
527 1b2afa91 2021-08-17 op 'help-echo
528 1b2afa91 2021-08-17 op (if isdir
529 3ec8a8b4 2021-08-17 op (concat
530 3ec8a8b4 2021-08-17 op "Directory\n"
531 3ec8a8b4 2021-08-17 op "VC operations can be applied to it\n"
532 3ec8a8b4 2021-08-17 op "mouse-3: Pop-up menu")
533 1b2afa91 2021-08-17 op "File\nmouse-3: Pop-up menu")
534 1b2afa91 2021-08-17 op 'mouse-face 'highlight
535 1b2afa91 2021-08-17 op 'keymap vc-dir-filename-mouse-map))))
536 1b2afa91 2021-08-17 op
537 af5ef7cd 2020-11-29 op (defun vc-got-working-revision (file)
538 3ec8a8b4 2021-08-17 op "Return the last commit that touched FILE or \"0\" if it's newly added."
539 af5ef7cd 2020-11-29 op (or
540 af5ef7cd 2020-11-29 op (with-temp-buffer
541 518ede14 2020-12-05 op (when (vc-got--log file 1)
542 af5ef7cd 2020-11-29 op (let (start)
543 af5ef7cd 2020-11-29 op (goto-char (point-min))
544 76d978fa 2021-01-05 op (forward-word) ; skip "commit"
545 76d978fa 2021-01-05 op (forward-char) ; skip the space
546 76d978fa 2021-01-05 op (setq start (point)) ; store start of the SHA
547 76d978fa 2021-01-05 op (forward-word) ; goto SHA end
548 af5ef7cd 2020-11-29 op (buffer-substring start (point)))))
549 af5ef7cd 2020-11-29 op ;; special case: if this file is added but has no previous commits
550 af5ef7cd 2020-11-29 op ;; touching it, got log will fail (as expected), but we have to
551 af5ef7cd 2020-11-29 op ;; return "0".
552 af5ef7cd 2020-11-29 op (when (eq (vc-got-state file) 'added)
553 af5ef7cd 2020-11-29 op "0")))
554 af5ef7cd 2020-11-29 op
555 af5ef7cd 2020-11-29 op (defun vc-got-checkout-model (_files)
556 01bf4e07 2021-08-17 op "Return the checkout model.
557 01bf4e07 2021-08-17 op Got uses an implicit checkout model for every file."
558 af5ef7cd 2020-11-29 op 'implicit)
559 af5ef7cd 2020-11-29 op
560 694534b4 2020-12-05 op (defun vc-got-mode-line-string (file)
561 694534b4 2020-12-05 op "Return the VC mode line string for FILE."
562 694534b4 2020-12-05 op (vc-got-with-worktree file
563 694534b4 2020-12-05 op (let ((def (vc-default-mode-line-string 'Got file)))
564 694534b4 2020-12-05 op (concat (substring def 0 4) (vc-got--current-branch)))))
565 694534b4 2020-12-05 op
566 23a0b465 2020-11-30 op
567 23a0b465 2020-11-30 op ;; state-changing functions
568 23a0b465 2020-11-30 op
569 23a0b465 2020-11-30 op (defun vc-got-create-repo (_backend)
570 7726b6c9 2021-08-10 op "Create an empty repository in the current directory."
571 7726b6c9 2021-08-10 op (error "[vc-got] create-repo not implemented"))
572 23a0b465 2020-11-30 op
573 23a0b465 2020-11-30 op (defun vc-got-register (files &optional _comment)
574 23a0b465 2020-11-30 op "Register FILES, passing `vc-register-switches' to the backend command."
575 23a0b465 2020-11-30 op (vc-got--add files))
576 23a0b465 2020-11-30 op
577 9e805da8 2020-11-30 op (defalias 'vc-got-responsible-p #'vc-got-root)
578 241a3b12 2021-08-11 op
579 241a3b12 2021-08-11 op (defun vc-got-unregister (file)
580 241a3b12 2021-08-11 op "Unregister FILE."
581 9114bbc3 2021-08-11 op (vc-got--remove file t t))
582 23a0b465 2020-11-30 op
583 23a0b465 2020-11-30 op (defun vc-got-checkin (files comment &optional _rev)
584 23a0b465 2020-11-30 op "Commit FILES with COMMENT as commit message."
585 23a0b465 2020-11-30 op (with-temp-buffer
586 ad36649b 2021-02-25 op (unless (zerop (vc-got--call "commit" "-m"
587 e7747ce0 2021-08-11 op (log-edit-extract-headers nil comment)
588 c3185bf7 2021-08-11 op "--"
589 ad36649b 2021-02-25 op files))
590 26362bf0 2021-08-11 op (error "[vc-got] can't commit: %s" (buffer-string)))))
591 23a0b465 2020-11-30 op
592 23a0b465 2020-11-30 op (defun vc-got-find-revision (file rev buffer)
593 c0c9a339 2020-12-04 op "Fill BUFFER with the content of FILE in the given revision REV."
594 961d35b5 2021-09-09 op (with-current-buffer buffer
595 961d35b5 2021-09-09 op (vc-got-with-worktree file
596 961d35b5 2021-09-09 op (vc-got--cat rev (file-relative-name file)))))
597 55091167 2020-12-05 op
598 eb85ad27 2020-12-05 op (defun vc-got-checkout (_file &optional _rev)
599 76d978fa 2021-01-05 op "Checkout revision REV of FILE.
600 76d978fa 2021-01-05 op If REV is t, checkout from the head."
601 7726b6c9 2021-08-10 op (error "[vc-got] checkout not implemented"))
602 eb85ad27 2020-12-05 op
603 eb85ad27 2020-12-05 op (defun vc-got-revert (file &optional _content-done)
604 eb85ad27 2020-12-05 op "Revert FILE back to working revision."
605 eb85ad27 2020-12-05 op (vc-got--revert file))
606 eb85ad27 2020-12-05 op
607 eb85ad27 2020-12-05 op (defun vc-got-merge-branch ()
608 eb85ad27 2020-12-05 op "Prompt for a branch and integrate it into the current one."
609 eb85ad27 2020-12-05 op ;; XXX: be smart and try to "got rebase" if "got integrate" fails?
610 eb85ad27 2020-12-05 op (let* ((branches (cl-loop for (branch . commit) in (vc-got--list-branches)
611 eb85ad27 2020-12-05 op collect branch))
612 eb85ad27 2020-12-05 op (branch (completing-read "Merge from branch: " branches)))
613 eb85ad27 2020-12-05 op (when branch
614 eb85ad27 2020-12-05 op (vc-got--integrate branch))))
615 686eac9a 2020-12-07 op
616 74b00009 2021-08-31 op (defun vc-got--proc-filter (proc s)
617 74b00009 2021-08-31 op "Custom output filter for async process PROC.
618 74b00009 2021-08-31 op It's like `vc-process-filter' but supports \r inside S."
619 74b00009 2021-08-31 op (let ((buffer (process-buffer proc)))
620 74b00009 2021-08-31 op (when (buffer-live-p buffer)
621 74b00009 2021-08-31 op (with-current-buffer buffer
622 74b00009 2021-08-31 op (save-excursion
623 74b00009 2021-08-31 op (let ((buffer-undo-list t)
624 74b00009 2021-08-31 op (inhibit-read-only t))
625 74b00009 2021-08-31 op (goto-char (process-mark proc))
626 74b00009 2021-08-31 op (if (not (string-match ".*\r\\(.*\\)" s))
627 74b00009 2021-08-31 op (insert s)
628 74b00009 2021-08-31 op ;; handle \r
629 74b00009 2021-08-31 op (end-of-line)
630 74b00009 2021-08-31 op (let ((end (point)))
631 74b00009 2021-08-31 op (beginning-of-line)
632 74b00009 2021-08-31 op (delete-region (point) end))
633 74b00009 2021-08-31 op (insert (match-string 1 s)))
634 74b00009 2021-08-31 op (set-marker (process-mark proc) (point))))))))
635 74b00009 2021-08-31 op
636 dec54a74 2021-02-21 op (defun vc-got--push-pull (cmd op prompt)
637 dec54a74 2021-02-21 op "Execute CMD OP, or prompt the user if PROMPT is non-nil."
638 dec54a74 2021-02-21 op (let ((buffer (format "*vc-got : %s*" (expand-file-name default-directory))))
639 686eac9a 2020-12-07 op (when-let (cmd (if prompt
640 686eac9a 2020-12-07 op (split-string
641 686eac9a 2020-12-07 op (read-shell-command (format "%s %s command: " cmd op)
642 1870c8bd 2021-03-27 op (format "%s %s " cmd op))
643 686eac9a 2020-12-07 op " " t)
644 686eac9a 2020-12-07 op (list cmd op)))
645 dec54a74 2021-02-21 op (apply #'vc-do-async-command buffer default-directory cmd)
646 dec54a74 2021-02-21 op (with-current-buffer buffer
647 74b00009 2021-08-31 op (vc-compilation-mode 'got)
648 74b00009 2021-08-31 op (let ((comp-cmd (mapconcat #'identity cmd " "))
649 74b00009 2021-08-31 op (proc (get-buffer-process buffer)))
650 95a78ecb 2021-08-31 op (setq-local compile-command comp-cmd)
651 95a78ecb 2021-08-31 op (setq-local compilation-directory default-directory)
652 95a78ecb 2021-08-31 op (setq-local compilation-arguments (list comp-cmd
653 dec54a74 2021-02-21 op nil
654 dec54a74 2021-02-21 op (lambda (_ign) buffer)
655 74b00009 2021-08-31 op nil))
656 74b00009 2021-08-31 op ;; Setup a custom process filter that handles \r.
657 74b00009 2021-08-31 op (set-process-filter proc #'vc-got--proc-filter)))
658 dec54a74 2021-02-21 op (vc-set-async-update buffer))))
659 eb85ad27 2020-12-05 op
660 3ec8a8b4 2021-08-17 op ;; TODO: this could be expanded. After a pull the worktree needs to
661 95782f5b 2022-02-01 op ;; be updated, either with a ``got update -b branch-name'' or ``got
662 95782f5b 2022-02-01 op ;; update -b remote/branchname'' plus a rebase.
663 eb85ad27 2020-12-05 op (defun vc-got-pull (prompt)
664 16a2f37b 2022-05-28 op "Execute a fetch prompting for the full command if PROMPT is not nil."
665 e4ac2ae5 2021-10-04 op (vc-got--push-pull vc-got-program "fetch" prompt))
666 686eac9a 2020-12-07 op
667 686eac9a 2020-12-07 op (defun vc-got-push (prompt)
668 16a2f37b 2022-05-28 op "Execute a send prompting for the full command if PROMPT is not nil."
669 e4ac2ae5 2021-10-04 op (vc-got--push-pull vc-got-program "send" prompt))
670 eb85ad27 2020-12-05 op
671 424f5152 2021-08-10 op
672 424f5152 2021-08-10 op ;; History functions
673 424f5152 2021-08-10 op
674 eb85ad27 2020-12-05 op (defun vc-got-print-log (files buffer &optional _shortlog start-revision limit)
675 eb85ad27 2020-12-05 op "Insert the revision log for FILES into BUFFER.
676 76d978fa 2021-01-05 op LIMIT limits the number of commits, optionally starting at
677 76d978fa 2021-01-05 op START-REVISION."
678 eb85ad27 2020-12-05 op (with-current-buffer buffer
679 eb85ad27 2020-12-05 op ;; the *vc-diff* may be read only
680 4571b1fd 2020-12-05 op (let ((inhibit-read-only t))
681 eb85ad27 2020-12-05 op (cl-loop for file in files
682 3ec8a8b4 2021-08-17 op do (vc-got--log (file-relative-name file)
683 3ec8a8b4 2021-08-17 op limit
684 3ec8a8b4 2021-08-17 op start-revision)))))
685 eb85ad27 2020-12-05 op
686 9ad1eb5d 2020-12-07 op (defun vc-got-log-outgoing (buffer remote-location)
687 9ad1eb5d 2020-12-07 op "Fill BUFFER with the diff between the local worktree branch and REMOTE-LOCATION."
688 9ad1eb5d 2020-12-07 op (vc-setup-buffer buffer)
689 a1839dca 2021-08-17 op (let ((rl (vc-got-next-revision
690 a1839dca 2021-08-17 op nil
691 a1839dca 2021-08-17 op (if (or (not remote-location) (string-empty-p remote-location))
692 a1839dca 2021-08-17 op (concat "origin/" (vc-got--current-branch))
693 a1839dca 2021-08-17 op remote-location)))
694 9ad1eb5d 2020-12-07 op (inhibit-read-only t))
695 9ad1eb5d 2020-12-07 op (with-current-buffer buffer
696 9ad1eb5d 2020-12-07 op (vc-got--log nil nil nil rl))))
697 6eda2f1f 2020-12-26 op
698 6eda2f1f 2020-12-26 op (defun vc-got-incoming (buffer remote-location)
699 3391eaf8 2022-02-01 op "Fill BUFFER with the incoming diff from REMOTE-LOCATION.
700 3391eaf8 2022-02-01 op That is, the diff between REMOTE-LOCATION and the local repository."
701 6eda2f1f 2020-12-26 op (let ((rl (if (or (not remote-location) (string-empty-p remote-location))
702 6eda2f1f 2020-12-26 op (concat "origin/" (vc-got--current-branch))
703 6eda2f1f 2020-12-26 op remote-location))
704 6eda2f1f 2020-12-26 op (inhibit-read-only t))
705 6eda2f1f 2020-12-26 op (with-current-buffer buffer
706 6eda2f1f 2020-12-26 op (vc-got--log nil nil (vc-got--current-branch) rl))))
707 9ad1eb5d 2020-12-07 op
708 eb85ad27 2020-12-05 op (defun vc-got-log-search (buffer pattern)
709 eb85ad27 2020-12-05 op "Search commits for PATTERN and write the results found in BUFFER."
710 eb85ad27 2020-12-05 op (with-current-buffer buffer
711 eb85ad27 2020-12-05 op (let ((inhibit-read-only t))
712 9ad1eb5d 2020-12-07 op (vc-got--log nil nil nil nil pattern))))
713 eb85ad27 2020-12-05 op
714 0a3316a6 2021-02-22 op (define-derived-mode vc-got-log-view-mode log-view-mode "Got-Log-View"
715 0a3316a6 2021-02-22 op "Got-specific log-view mode.
716 0a3316a6 2021-02-22 op Heavily inspired by `vc-git-log-view-mode'."
717 0a3316a6 2021-02-22 op (require 'add-log)
718 95a78ecb 2021-08-31 op (setq-local log-view-file-re regexp-unmatchable)
719 95a78ecb 2021-08-31 op (setq-local log-view-per-file-logs nil)
720 95a78ecb 2021-08-31 op (setq-local log-view-message-re "^commit +\\([0-9a-z]+\\)")
721 95a78ecb 2021-08-31 op (setq-local log-view-font-lock-keywords
722 95a78ecb 2021-08-31 op (append
723 95a78ecb 2021-08-31 op `((,log-view-message-re (1 'change-log-acknowledgment)))
724 95a78ecb 2021-08-31 op ;; Handle the case:
725 95a78ecb 2021-08-31 op ;; user: foo@bar
726 95a78ecb 2021-08-31 op '(("^from: \\([A-Za-z0-9_.+-]+@[A-Za-z0-9_.-]+\\)"
727 95a78ecb 2021-08-31 op (1 'change-log-email))
728 95a78ecb 2021-08-31 op ;; Handle the case:
729 95a78ecb 2021-08-31 op ;; user: FirstName LastName <foo@bar>
730 95a78ecb 2021-08-31 op ("^from: \\([^<(]+?\\)[ \t]*[(<]\\([A-Za-z0-9_.+-]+@[A-Za-z0-9_.-]+\\)[>)]"
731 95a78ecb 2021-08-31 op (1 'change-log-name)
732 95a78ecb 2021-08-31 op (2 'change-log-email))
733 95a78ecb 2021-08-31 op ("^date: \\(.+\\)" (1 'change-log-date))))))
734 0a3316a6 2021-02-22 op
735 eb85ad27 2020-12-05 op ;; TODO: async
736 f18d3e11 2020-12-05 op ;; TODO: return 0 or 1
737 eb85ad27 2020-12-05 op (defun vc-got-diff (files &optional rev1 rev2 buffer _async)
738 eb85ad27 2020-12-05 op "Insert into BUFFER (or *vc-diff*) the diff for FILES from REV1 to REV2."
739 f457868b 2020-12-07 op (let* ((buffer (get-buffer-create (or buffer "*vc-diff*")))
740 eb85ad27 2020-12-05 op (inhibit-read-only t))
741 eb85ad27 2020-12-05 op (with-current-buffer buffer
742 8ec0bb76 2021-08-11 op (vc-got-with-worktree (or (car files)
743 8ec0bb76 2021-08-11 op default-directory)
744 32819756 2021-09-14 op (cond ((and (null rev1)
745 32819756 2021-09-14 op (null rev2))
746 45f7f941 2021-11-09 op (vc-got--diff-files files))
747 32819756 2021-09-14 op ((and (null rev1)
748 32819756 2021-09-14 op rev2)
749 32819756 2021-09-14 op ;; TODO: this includes the whole diff while to respect
750 32819756 2021-09-14 op ;; the vc semantics we should filter only the diff for
751 32819756 2021-09-14 op ;; files in FILES.
752 32819756 2021-09-14 op ;;
753 32819756 2021-09-14 op ;; XXX: this includes also the commit message, I
754 32819756 2021-09-14 op ;; consider it a feature over the usual vc behaviour of
755 32819756 2021-09-14 op ;; showing only the diff.
756 32819756 2021-09-14 op (vc-got--log nil 1 rev2 nil nil nil t))
757 32819756 2021-09-14 op ;;
758 32819756 2021-09-14 op ;; TODO: if rev1 is nil, diff from the current version until
759 32819756 2021-09-14 op ;; rev2.
760 32819756 2021-09-14 op ;;
761 32819756 2021-09-14 op ;; TODO 2: if rev2 is nil as well, diff against an empty
762 32819756 2021-09-14 op ;; tree (i.e. get the patch from `got log -p rev1')
763 36e5655f 2021-11-09 op (t (vc-got--diff-objects rev1 rev2)))))))
764 5fb2f474 2021-08-10 op
765 5fb2f474 2021-08-10 op (defun vc-got-revision-completion-table (_files)
766 5fb2f474 2021-08-10 op "Return a completion table for existing revisions.
767 5fb2f474 2021-08-10 op Ignores FILES because GoT doesn't have the concept of ``file
768 5fb2f474 2021-08-10 op revisions''; instead, like with git, you have tags and branches."
769 5fb2f474 2021-08-10 op (letrec ((table (lazy-completion-table
770 5fb2f474 2021-08-10 op table (lambda () (vc-got--ref)))))
771 5fb2f474 2021-08-10 op table))
772 3f25d9ff 2020-01-02 noreply
773 3f25d9ff 2020-01-02 noreply (defun vc-got-annotate-command (file buf &optional rev)
774 67332bad 2021-01-05 op "Show annotated contents of FILE in buffer BUF. If given, use revision REV."
775 3f25d9ff 2020-01-02 noreply (let (process-file-side-effects)
776 3f25d9ff 2020-01-02 noreply (with-current-buffer buf
777 3f25d9ff 2020-01-02 noreply ;; FIXME: vc-ensure-vc-buffer won't recognise this buffer as managed
778 3f25d9ff 2020-01-02 noreply ;; by got unless vc-parent-buffer points to a buffer managed by got.
779 3f25d9ff 2020-01-02 noreply ;; investigate why this is needed.
780 3efc7598 2021-09-09 op (setq-local vc-parent-buffer (find-file-noselect file))
781 76d978fa 2021-01-05 op (vc-got--call "blame"
782 76d978fa 2021-01-05 op (when rev (list "-c" rev))
783 c3185bf7 2021-08-11 op "--"
784 76d978fa 2021-01-05 op file))))
785 3f25d9ff 2020-01-02 noreply
786 3f25d9ff 2020-01-02 noreply (defconst vc-got--annotate-re
787 3f25d9ff 2020-01-02 noreply (concat "^[0-9]\\{1,\\}) " ; line number followed by )
788 3f25d9ff 2020-01-02 noreply "\\([a-z0-9]+\\) " ; SHA-1 of commit
789 3f25d9ff 2020-01-02 noreply "\\([0-9]\\{4\\}-[0-9]\\{2\\}-[0-9]\\{2\\}\\) " ; year-mm-dd
790 3f25d9ff 2020-01-02 noreply "\\([^ ]\\)+ ") ; author
791 3f25d9ff 2020-01-02 noreply "Regexp to match annotation output lines.
792 3f25d9ff 2020-01-02 noreply Provides capture groups for:
793 3f25d9ff 2020-01-02 noreply 1. revision id
794 3f25d9ff 2020-01-02 noreply 2. date of commit
795 3f25d9ff 2020-01-02 noreply 3. author of commit")
796 3f25d9ff 2020-01-02 noreply
797 3f25d9ff 2020-01-02 noreply (defconst vc-got--commit-re "^commit \\([a-z0-9]+\\)"
798 3f25d9ff 2020-01-02 noreply "Regexp to match commit lines.
799 3f25d9ff 2020-01-02 noreply Provides capture group for the commit revision id.")
800 3f25d9ff 2020-01-02 noreply
801 3f25d9ff 2020-01-02 noreply (defun vc-got-annotate-time ()
802 3f25d9ff 2020-01-02 noreply "Return the time of the next line of annotation at or after point.
803 3f25d9ff 2020-01-02 noreply Value is returned as floating point fractional number of days."
804 01c0cd8c 2021-09-14 op ;; XXX: to behave like vc-git here we should call re-search-forward
805 01c0cd8c 2021-09-14 op ;; instead of looking-at, as it makes the fontification of the line
806 01c0cd8c 2021-09-14 op ;; start AFTER the info. The problem is, due to the format of the
807 01c0cd8c 2021-09-14 op ;; blame, it produces an ugly result, with colors starting at
808 0a5a024b 2022-02-01 op ;; different offsets depending on how long the committer name is.
809 01c0cd8c 2021-09-14 op (when (looking-at vc-got--annotate-re)
810 01c0cd8c 2021-09-14 op (let ((str (match-string-no-properties 2)))
811 01c0cd8c 2021-09-14 op (vc-annotate-convert-time
812 01c0cd8c 2021-09-14 op (encode-time 0 0 0
813 01c0cd8c 2021-09-14 op (string-to-number (substring str 8 10))
814 01c0cd8c 2021-09-14 op (string-to-number (substring str 5 7))
815 01c0cd8c 2021-09-14 op (string-to-number (substring str 0 4)))))))
816 eb85ad27 2020-12-05 op
817 3f25d9ff 2020-01-02 noreply (defun vc-got-annotate-extract-revision-at-line ()
818 67332bad 2021-01-05 op "Return revision corresponding to the current line or nil."
819 3f25d9ff 2020-01-02 noreply (save-excursion
820 3f25d9ff 2020-01-02 noreply (beginning-of-line)
821 3f25d9ff 2020-01-02 noreply (when (looking-at vc-got--annotate-re)
822 3f25d9ff 2020-01-02 noreply (match-string-no-properties 1))))
823 3f25d9ff 2020-01-02 noreply
824 424f5152 2021-08-10 op
825 8014bde1 2021-08-10 op ;; Tag system
826 6ab1c7b4 2021-08-11 op
827 6ab1c7b4 2021-08-11 op (defun vc-got--tag-callback (tag)
828 6ab1c7b4 2021-08-11 op "`log-edit' callback for `vc-got-create-tag'.
829 6ab1c7b4 2021-08-11 op Creates the TAG using the content of the current buffer."
830 6ab1c7b4 2021-08-11 op (interactive)
831 6ab1c7b4 2021-08-11 op (let ((msg (buffer-substring-no-properties (point-min)
832 6ab1c7b4 2021-08-11 op (point-max))))
833 6ab1c7b4 2021-08-11 op (with-temp-buffer
834 3dcaa71b 2021-08-31 op (unless (zerop (vc-got--call "tag"
835 3dcaa71b 2021-08-31 op "-m"
836 3dcaa71b 2021-08-31 op (log-edit-extract-headers nil msg)
837 3dcaa71b 2021-08-31 op "--"
838 3dcaa71b 2021-08-31 op tag))
839 6ab1c7b4 2021-08-11 op (error "[vc-got] can't create tag %s: %s" tag (buffer-string))))))
840 8014bde1 2021-08-10 op
841 8014bde1 2021-08-10 op (defun vc-got-create-tag (_dir name branchp)
842 8014bde1 2021-08-10 op "Attach the tag NAME to the state of the worktree.
843 6ab1c7b4 2021-08-11 op DIR is ignored (tags are global, not per-file). If BRANCHP is
844 6ab1c7b4 2021-08-11 op true, NAME should create a new branch otherwise it will pop-up a
845 6ab1c7b4 2021-08-11 op `log-edit' buffer to provide the tag message."
846 8014bde1 2021-08-10 op ;; TODO: vc reccomends to ensure that all the file are in a clean
847 8014bde1 2021-08-10 op ;; state, but is it useful?
848 8014bde1 2021-08-10 op (if branchp
849 8014bde1 2021-08-10 op (vc-got--branch name)
850 6ab1c7b4 2021-08-11 op (let ((buf (get-buffer-create "*vc-got tag*")))
851 6ab1c7b4 2021-08-11 op (with-current-buffer buf
852 6ab1c7b4 2021-08-11 op (erase-buffer)
853 3dcaa71b 2021-08-31 op (save-excursion
854 3dcaa71b 2021-08-31 op (insert "Summary: tag " name "\n\n"))
855 3dcaa71b 2021-08-31 op (move-end-of-line 1)
856 6ab1c7b4 2021-08-11 op (switch-to-buffer buf)
857 6ab1c7b4 2021-08-11 op (log-edit (lambda ()
858 6ab1c7b4 2021-08-11 op (interactive)
859 6ab1c7b4 2021-08-11 op (unwind-protect
860 6ab1c7b4 2021-08-11 op (vc-got--tag-callback name)
861 6ab1c7b4 2021-08-11 op (kill-buffer buf))))))))
862 8014bde1 2021-08-10 op
863 09d59379 2021-08-11 op (defun vc-got-retrieve-tag (dir name _update)
864 09d59379 2021-08-11 op "Switch to the tag NAME for files at or below DIR."
865 09d59379 2021-08-11 op (let ((default-directory dir))
866 db89d6fd 2021-08-16 op (vc-got--update name dir)))
867 09d59379 2021-08-11 op
868 8014bde1 2021-08-10 op
869 424f5152 2021-08-10 op ;; Miscellaneous
870 424f5152 2021-08-10 op
871 2552a6c7 2021-08-11 op (defun vc-got-find-ignore-file (file)
872 2552a6c7 2021-08-11 op "Return the gitignore file that controls FILE."
873 2552a6c7 2021-08-11 op (expand-file-name ".gitignore"
874 2552a6c7 2021-08-11 op (vc-got-root file)))
875 2552a6c7 2021-08-11 op
876 3f25d9ff 2020-01-02 noreply (defun vc-got-previous-revision (file rev)
877 3ec8a8b4 2021-08-17 op "Return the revision number that precedes REV for FILE or nil."
878 3f25d9ff 2020-01-02 noreply (with-temp-buffer
879 3f25d9ff 2020-01-02 noreply (vc-got--log file 2 rev nil nil t)
880 3f25d9ff 2020-01-02 noreply (goto-char (point-min))
881 3f25d9ff 2020-01-02 noreply (keep-lines "^commit")
882 3f25d9ff 2020-01-02 noreply (when (looking-at vc-got--commit-re)
883 3f25d9ff 2020-01-02 noreply (match-string-no-properties 1))))
884 3f25d9ff 2020-01-02 noreply
885 3f25d9ff 2020-01-02 noreply (defun vc-got-next-revision (file rev)
886 3ec8a8b4 2021-08-17 op "Return the revision number that follows REV for FILE or nil."
887 3f25d9ff 2020-01-02 noreply (with-temp-buffer
888 3f25d9ff 2020-01-02 noreply (vc-got--log file nil nil rev)
889 3f25d9ff 2020-01-02 noreply (keep-lines "^commit" (point-min) (point-max))
890 3f25d9ff 2020-01-02 noreply (goto-char (point-max))
891 76d978fa 2021-01-05 op (forward-line -1) ; return from empty line to last actual commit
892 3f25d9ff 2020-01-02 noreply (unless (= (point) (point-min))
893 3f25d9ff 2020-01-02 noreply (forward-line -1)
894 3f25d9ff 2020-01-02 noreply (when (looking-at vc-got--commit-re)
895 3f25d9ff 2020-01-02 noreply (match-string-no-properties 1)))))
896 3f25d9ff 2020-01-02 noreply
897 a96df0e8 2020-01-03 noreply (defun vc-got-delete-file (file)
898 a96df0e8 2020-01-03 noreply "Delete FILE locally and mark it deleted in work tree."
899 a96df0e8 2020-01-03 noreply (vc-got--remove file t))
900 a96df0e8 2020-01-03 noreply
901 c151c257 2021-08-03 op (defun vc-got-find-file-hook ()
902 c151c257 2021-08-03 op "Activate `smerge-mode' if there is a conflict."
903 c151c257 2021-08-03 op ;; just like vc-git-find-file-hook
904 c151c257 2021-08-03 op (when (and buffer-file-name
905 c151c257 2021-08-03 op (eq (vc-state buffer-file-name 'Got) 'conflict)
906 c151c257 2021-08-03 op (save-excursion
907 c151c257 2021-08-03 op (goto-char (point-min))
908 c151c257 2021-08-03 op (re-search-forward "^<<<<<<< " nil 'noerror)))
909 c151c257 2021-08-03 op (smerge-start-session)
910 c151c257 2021-08-03 op (vc-message-unresolved-conflicts buffer-file-name)))
911 c151c257 2021-08-03 op
912 7bb16f79 2020-01-03 op (defun vc-got-conflicted-files (dir)
913 7bb16f79 2020-01-03 op "Return the list of files with conflicts in directory DIR."
914 7bb16f79 2020-01-03 op (let* ((root (vc-got-root dir))
915 7bb16f79 2020-01-03 op (default-directory root)
916 3efc7598 2021-09-09 op (process-file-side-effects nil))
917 3efc7598 2021-09-09 op (cl-loop for (file status _) in (vc-got--status "C" ".")
918 3efc7598 2021-09-09 op when (and (eq status 'conflict)
919 3efc7598 2021-09-09 op (file-in-directory-p file dir))
920 3efc7598 2021-09-09 op collect file)))
921 7bb16f79 2020-01-03 op
922 8b635a9f 2020-01-03 op (defun vc-got-repository-url (_file &optional remote-name)
923 8b635a9f 2020-01-03 op "Return URL for REMOTE-NAME, or for \"origin\" if nil."
924 8b635a9f 2020-01-03 op (let* ((default-directory (vc-got--repo-root))
925 8b635a9f 2020-01-03 op (remote-name (or remote-name "origin"))
926 8b635a9f 2020-01-03 op (heading (concat "[remote \"" remote-name "\"]"))
927 76d978fa 2021-01-05 op (conf (cond ((file-exists-p ".git/config") ".git/config")
928 76d978fa 2021-01-05 op ((file-exists-p ".git") nil)
929 76d978fa 2021-01-05 op ((file-exists-p "config") "config")))
930 8b635a9f 2020-01-03 op found)
931 6f42ede4 2021-01-05 op (when conf
932 6f42ede4 2021-01-05 op (with-temp-buffer
933 f8a9db56 2020-01-03 op (insert-file-contents conf)
934 f8a9db56 2020-01-03 op (goto-char (point-min))
935 f8a9db56 2020-01-03 op (when (search-forward heading nil t)
936 f8a9db56 2020-01-03 op (forward-line)
937 f8a9db56 2020-01-03 op (while (and (not found)
938 76d978fa 2021-01-05 op (looking-at ".*=") ; too broad?
939 7a5ead65 2020-01-03 op (not (= (point) (point-max))))
940 f8a9db56 2020-01-03 op (when (looking-at ".*url = \\(.*\\)")
941 7a5ead65 2020-01-03 op (setq found (match-string-no-properties 1)))
942 7a5ead65 2020-01-03 op (forward-line))
943 f8a9db56 2020-01-03 op found)))))
944 a1679b13 2021-09-09 op
945 a1679b13 2021-09-09 op
946 8b635a9f 2020-01-03 op
947 a1679b13 2021-09-09 op ;; Automatically register the backend and add ".got" to the exclusion
948 a1679b13 2021-09-09 op ;; list.
949 a1679b13 2021-09-09 op
950 a1679b13 2021-09-09 op ;;;###autoload
951 a1679b13 2021-09-09 op (add-to-list 'vc-handled-backends 'Got)
952 a1679b13 2021-09-09 op
953 a1679b13 2021-09-09 op ;;;###autoload
954 a1679b13 2021-09-09 op (add-to-list 'vc-directory-exclusion-list ".got")
955 a1679b13 2021-09-09 op
956 af5ef7cd 2020-11-29 op (provide 'vc-got)
957 af5ef7cd 2020-11-29 op ;;; vc-got.el ends here