Blame


1 3d1b16d1 2023-07-08 jrick /*
2 3d1b16d1 2023-07-08 jrick * Copyright (c) 2018, 2019, 2020 Stefan Sperling <stsp@openbsd.org>
3 3d1b16d1 2023-07-08 jrick *
4 3d1b16d1 2023-07-08 jrick * Permission to use, copy, modify, and distribute this software for any
5 3d1b16d1 2023-07-08 jrick * purpose with or without fee is hereby granted, provided that the above
6 3d1b16d1 2023-07-08 jrick * copyright notice and this permission notice appear in all copies.
7 3d1b16d1 2023-07-08 jrick *
8 3d1b16d1 2023-07-08 jrick * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
9 3d1b16d1 2023-07-08 jrick * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
10 3d1b16d1 2023-07-08 jrick * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
11 3d1b16d1 2023-07-08 jrick * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
12 3d1b16d1 2023-07-08 jrick * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
13 3d1b16d1 2023-07-08 jrick * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
14 3d1b16d1 2023-07-08 jrick * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
15 3d1b16d1 2023-07-08 jrick */
16 3d1b16d1 2023-07-08 jrick
17 3d1b16d1 2023-07-08 jrick #include <sys/stat.h>
18 3d1b16d1 2023-07-08 jrick #include <sys/queue.h>
19 3d1b16d1 2023-07-08 jrick #include <sys/tree.h>
20 3d1b16d1 2023-07-08 jrick
21 3d1b16d1 2023-07-08 jrick #include <dirent.h>
22 3d1b16d1 2023-07-08 jrick #include <limits.h>
23 3d1b16d1 2023-07-08 jrick #include <stddef.h>
24 3d1b16d1 2023-07-08 jrick #include <string.h>
25 3d1b16d1 2023-07-08 jrick #include <stdio.h>
26 3d1b16d1 2023-07-08 jrick #include <stdlib.h>
27 3d1b16d1 2023-07-08 jrick #include <time.h>
28 3d1b16d1 2023-07-08 jrick #include <fcntl.h>
29 3d1b16d1 2023-07-08 jrick #include <errno.h>
30 3d1b16d1 2023-07-08 jrick #include <unistd.h>
31 3d1b16d1 2023-07-08 jrick #include <sha1.h>
32 3d1b16d1 2023-07-08 jrick #include <sha2.h>
33 3d1b16d1 2023-07-08 jrick #include <zlib.h>
34 3d1b16d1 2023-07-08 jrick #include <fnmatch.h>
35 3d1b16d1 2023-07-08 jrick #include <libgen.h>
36 3d1b16d1 2023-07-08 jrick #include <uuid.h>
37 3d1b16d1 2023-07-08 jrick #include <util.h>
38 3d1b16d1 2023-07-08 jrick
39 3d1b16d1 2023-07-08 jrick #include "got_error.h"
40 3d1b16d1 2023-07-08 jrick #include "got_repository.h"
41 3d1b16d1 2023-07-08 jrick #include "got_reference.h"
42 3d1b16d1 2023-07-08 jrick #include "got_object.h"
43 3d1b16d1 2023-07-08 jrick #include "got_path.h"
44 3d1b16d1 2023-07-08 jrick #include "got_cancel.h"
45 3d1b16d1 2023-07-08 jrick #include "got_worktree.h"
46 3d1b16d1 2023-07-08 jrick #include "got_worktree_cvg.h"
47 3d1b16d1 2023-07-08 jrick #include "got_opentemp.h"
48 3d1b16d1 2023-07-08 jrick #include "got_diff.h"
49 3d1b16d1 2023-07-08 jrick #include "got_send.h"
50 3d1b16d1 2023-07-08 jrick #include "got_fetch.h"
51 3d1b16d1 2023-07-08 jrick
52 3d1b16d1 2023-07-08 jrick #include "got_lib_worktree.h"
53 3d1b16d1 2023-07-08 jrick #include "got_lib_hash.h"
54 3d1b16d1 2023-07-08 jrick #include "got_lib_fileindex.h"
55 3d1b16d1 2023-07-08 jrick #include "got_lib_inflate.h"
56 3d1b16d1 2023-07-08 jrick #include "got_lib_delta.h"
57 3d1b16d1 2023-07-08 jrick #include "got_lib_object.h"
58 3d1b16d1 2023-07-08 jrick #include "got_lib_object_parse.h"
59 3d1b16d1 2023-07-08 jrick #include "got_lib_object_create.h"
60 3d1b16d1 2023-07-08 jrick #include "got_lib_object_idset.h"
61 3d1b16d1 2023-07-08 jrick #include "got_lib_diff.h"
62 3d1b16d1 2023-07-08 jrick
63 3d1b16d1 2023-07-08 jrick #ifndef MIN
64 3d1b16d1 2023-07-08 jrick #define MIN(_a,_b) ((_a) < (_b) ? (_a) : (_b))
65 3d1b16d1 2023-07-08 jrick #endif
66 3d1b16d1 2023-07-08 jrick
67 3d1b16d1 2023-07-08 jrick #define GOT_MERGE_LABEL_MERGED "merged change"
68 3d1b16d1 2023-07-08 jrick #define GOT_MERGE_LABEL_BASE "3-way merge base"
69 3d1b16d1 2023-07-08 jrick
70 3d1b16d1 2023-07-08 jrick static const struct got_error *
71 3d1b16d1 2023-07-08 jrick lock_worktree(struct got_worktree *worktree, int operation)
72 3d1b16d1 2023-07-08 jrick {
73 3d1b16d1 2023-07-08 jrick if (flock(worktree->lockfd, operation | LOCK_NB) == -1)
74 3d1b16d1 2023-07-08 jrick return (errno == EWOULDBLOCK ? got_error(GOT_ERR_WORKTREE_BUSY)
75 3d1b16d1 2023-07-08 jrick : got_error_from_errno2("flock",
76 3d1b16d1 2023-07-08 jrick got_worktree_get_root_path(worktree)));
77 3d1b16d1 2023-07-08 jrick return NULL;
78 3d1b16d1 2023-07-08 jrick }
79 3d1b16d1 2023-07-08 jrick
80 3d1b16d1 2023-07-08 jrick static const struct got_error *
81 3d1b16d1 2023-07-08 jrick is_bad_symlink_target(int *is_bad_symlink, const char *target_path,
82 3d1b16d1 2023-07-08 jrick size_t target_len, const char *ondisk_path, const char *wtroot_path)
83 3d1b16d1 2023-07-08 jrick {
84 3d1b16d1 2023-07-08 jrick const struct got_error *err = NULL;
85 3d1b16d1 2023-07-08 jrick char canonpath[PATH_MAX];
86 3d1b16d1 2023-07-08 jrick char *path_got = NULL;
87 3d1b16d1 2023-07-08 jrick
88 3d1b16d1 2023-07-08 jrick *is_bad_symlink = 0;
89 3d1b16d1 2023-07-08 jrick
90 3d1b16d1 2023-07-08 jrick if (target_len >= sizeof(canonpath)) {
91 3d1b16d1 2023-07-08 jrick *is_bad_symlink = 1;
92 3d1b16d1 2023-07-08 jrick return NULL;
93 3d1b16d1 2023-07-08 jrick }
94 3d1b16d1 2023-07-08 jrick
95 3d1b16d1 2023-07-08 jrick /*
96 3d1b16d1 2023-07-08 jrick * We do not use realpath(3) to resolve the symlink's target
97 3d1b16d1 2023-07-08 jrick * path because we don't want to resolve symlinks recursively.
98 3d1b16d1 2023-07-08 jrick * Instead we make the path absolute and then canonicalize it.
99 3d1b16d1 2023-07-08 jrick * Relative symlink target lookup should begin at the directory
100 3d1b16d1 2023-07-08 jrick * in which the blob object is being installed.
101 3d1b16d1 2023-07-08 jrick */
102 3d1b16d1 2023-07-08 jrick if (!got_path_is_absolute(target_path)) {
103 3d1b16d1 2023-07-08 jrick char *abspath, *parent;
104 3d1b16d1 2023-07-08 jrick err = got_path_dirname(&parent, ondisk_path);
105 3d1b16d1 2023-07-08 jrick if (err)
106 3d1b16d1 2023-07-08 jrick return err;
107 3d1b16d1 2023-07-08 jrick if (asprintf(&abspath, "%s/%s", parent, target_path) == -1) {
108 3d1b16d1 2023-07-08 jrick free(parent);
109 3d1b16d1 2023-07-08 jrick return got_error_from_errno("asprintf");
110 3d1b16d1 2023-07-08 jrick }
111 3d1b16d1 2023-07-08 jrick free(parent);
112 3d1b16d1 2023-07-08 jrick if (strlen(abspath) >= sizeof(canonpath)) {
113 3d1b16d1 2023-07-08 jrick err = got_error_path(abspath, GOT_ERR_BAD_PATH);
114 3d1b16d1 2023-07-08 jrick free(abspath);
115 3d1b16d1 2023-07-08 jrick return err;
116 3d1b16d1 2023-07-08 jrick }
117 3d1b16d1 2023-07-08 jrick err = got_canonpath(abspath, canonpath, sizeof(canonpath));
118 3d1b16d1 2023-07-08 jrick free(abspath);
119 3d1b16d1 2023-07-08 jrick if (err)
120 3d1b16d1 2023-07-08 jrick return err;
121 3d1b16d1 2023-07-08 jrick } else {
122 3d1b16d1 2023-07-08 jrick err = got_canonpath(target_path, canonpath, sizeof(canonpath));
123 3d1b16d1 2023-07-08 jrick if (err)
124 3d1b16d1 2023-07-08 jrick return err;
125 3d1b16d1 2023-07-08 jrick }
126 3d1b16d1 2023-07-08 jrick
127 3d1b16d1 2023-07-08 jrick /* Only allow symlinks pointing at paths within the work tree. */
128 3d1b16d1 2023-07-08 jrick if (!got_path_is_child(canonpath, wtroot_path, strlen(wtroot_path))) {
129 3d1b16d1 2023-07-08 jrick *is_bad_symlink = 1;
130 3d1b16d1 2023-07-08 jrick return NULL;
131 3d1b16d1 2023-07-08 jrick }
132 3d1b16d1 2023-07-08 jrick
133 3d1b16d1 2023-07-08 jrick /* Do not allow symlinks pointing into the .got directory. */
134 3d1b16d1 2023-07-08 jrick if (asprintf(&path_got, "%s/%s", wtroot_path,
135 3d1b16d1 2023-07-08 jrick GOT_WORKTREE_GOT_DIR) == -1)
136 3d1b16d1 2023-07-08 jrick return got_error_from_errno("asprintf");
137 3d1b16d1 2023-07-08 jrick if (got_path_is_child(canonpath, path_got, strlen(path_got)))
138 3d1b16d1 2023-07-08 jrick *is_bad_symlink = 1;
139 3d1b16d1 2023-07-08 jrick
140 3d1b16d1 2023-07-08 jrick free(path_got);
141 3d1b16d1 2023-07-08 jrick return NULL;
142 3d1b16d1 2023-07-08 jrick }
143 3d1b16d1 2023-07-08 jrick
144 3d1b16d1 2023-07-08 jrick /*
145 3d1b16d1 2023-07-08 jrick * Upgrade STATUS_MODIFY to STATUS_CONFLICT if a
146 3d1b16d1 2023-07-08 jrick * conflict marker is found in newly added lines only.
147 3d1b16d1 2023-07-08 jrick */
148 3d1b16d1 2023-07-08 jrick static const struct got_error *
149 3d1b16d1 2023-07-08 jrick get_modified_file_content_status(unsigned char *status,
150 3d1b16d1 2023-07-08 jrick struct got_blob_object *blob, const char *path, struct stat *sb,
151 3d1b16d1 2023-07-08 jrick FILE *ondisk_file)
152 3d1b16d1 2023-07-08 jrick {
153 3d1b16d1 2023-07-08 jrick const struct got_error *err, *free_err;
154 3d1b16d1 2023-07-08 jrick const char *markers[3] = {
155 3d1b16d1 2023-07-08 jrick GOT_DIFF_CONFLICT_MARKER_BEGIN,
156 3d1b16d1 2023-07-08 jrick GOT_DIFF_CONFLICT_MARKER_SEP,
157 3d1b16d1 2023-07-08 jrick GOT_DIFF_CONFLICT_MARKER_END
158 3d1b16d1 2023-07-08 jrick };
159 3d1b16d1 2023-07-08 jrick FILE *f1 = NULL;
160 3d1b16d1 2023-07-08 jrick struct got_diffreg_result *diffreg_result = NULL;
161 3d1b16d1 2023-07-08 jrick struct diff_result *r;
162 3d1b16d1 2023-07-08 jrick int nchunks_parsed, n, i = 0, ln = 0;
163 3d1b16d1 2023-07-08 jrick char *line = NULL;
164 3d1b16d1 2023-07-08 jrick size_t linesize = 0;
165 3d1b16d1 2023-07-08 jrick ssize_t linelen;
166 3d1b16d1 2023-07-08 jrick
167 3d1b16d1 2023-07-08 jrick if (*status != GOT_STATUS_MODIFY)
168 3d1b16d1 2023-07-08 jrick return NULL;
169 3d1b16d1 2023-07-08 jrick
170 3d1b16d1 2023-07-08 jrick f1 = got_opentemp();
171 3d1b16d1 2023-07-08 jrick if (f1 == NULL)
172 3d1b16d1 2023-07-08 jrick return got_error_from_errno("got_opentemp");
173 3d1b16d1 2023-07-08 jrick
174 3d1b16d1 2023-07-08 jrick if (blob) {
175 3d1b16d1 2023-07-08 jrick got_object_blob_rewind(blob);
176 3d1b16d1 2023-07-08 jrick err = got_object_blob_dump_to_file(NULL, NULL, NULL, f1, blob);
177 3d1b16d1 2023-07-08 jrick if (err)
178 3d1b16d1 2023-07-08 jrick goto done;
179 3d1b16d1 2023-07-08 jrick }
180 3d1b16d1 2023-07-08 jrick
181 3d1b16d1 2023-07-08 jrick err = got_diff_files(&diffreg_result, f1, 1, NULL, ondisk_file,
182 3d1b16d1 2023-07-08 jrick 1, NULL, 0, 0, 1, NULL, GOT_DIFF_ALGORITHM_MYERS);
183 3d1b16d1 2023-07-08 jrick if (err)
184 3d1b16d1 2023-07-08 jrick goto done;
185 3d1b16d1 2023-07-08 jrick
186 3d1b16d1 2023-07-08 jrick r = diffreg_result->result;
187 3d1b16d1 2023-07-08 jrick
188 3d1b16d1 2023-07-08 jrick for (n = 0; n < r->chunks.len; n += nchunks_parsed) {
189 3d1b16d1 2023-07-08 jrick struct diff_chunk *c;
190 3d1b16d1 2023-07-08 jrick struct diff_chunk_context cc = {};
191 3d1b16d1 2023-07-08 jrick off_t pos;
192 3d1b16d1 2023-07-08 jrick
193 3d1b16d1 2023-07-08 jrick /*
194 3d1b16d1 2023-07-08 jrick * We can optimise a little by advancing straight
195 3d1b16d1 2023-07-08 jrick * to the next chunk if this one has no added lines.
196 3d1b16d1 2023-07-08 jrick */
197 3d1b16d1 2023-07-08 jrick c = diff_chunk_get(r, n);
198 3d1b16d1 2023-07-08 jrick
199 3d1b16d1 2023-07-08 jrick if (diff_chunk_type(c) != CHUNK_PLUS) {
200 3d1b16d1 2023-07-08 jrick nchunks_parsed = 1;
201 3d1b16d1 2023-07-08 jrick continue; /* removed or unchanged lines */
202 3d1b16d1 2023-07-08 jrick }
203 3d1b16d1 2023-07-08 jrick
204 3d1b16d1 2023-07-08 jrick pos = diff_chunk_get_right_start_pos(c);
205 3d1b16d1 2023-07-08 jrick if (fseek(ondisk_file, pos, SEEK_SET) == -1) {
206 3d1b16d1 2023-07-08 jrick err = got_ferror(ondisk_file, GOT_ERR_IO);
207 3d1b16d1 2023-07-08 jrick goto done;
208 3d1b16d1 2023-07-08 jrick }
209 3d1b16d1 2023-07-08 jrick
210 3d1b16d1 2023-07-08 jrick diff_chunk_context_load_change(&cc, &nchunks_parsed, r, n, 0);
211 3d1b16d1 2023-07-08 jrick ln = cc.right.start;
212 3d1b16d1 2023-07-08 jrick
213 3d1b16d1 2023-07-08 jrick while (ln < cc.right.end) {
214 3d1b16d1 2023-07-08 jrick linelen = getline(&line, &linesize, ondisk_file);
215 3d1b16d1 2023-07-08 jrick if (linelen == -1) {
216 3d1b16d1 2023-07-08 jrick if (feof(ondisk_file))
217 3d1b16d1 2023-07-08 jrick break;
218 3d1b16d1 2023-07-08 jrick err = got_ferror(ondisk_file, GOT_ERR_IO);
219 3d1b16d1 2023-07-08 jrick break;
220 3d1b16d1 2023-07-08 jrick }
221 3d1b16d1 2023-07-08 jrick
222 3d1b16d1 2023-07-08 jrick if (line && strncmp(line, markers[i],
223 3d1b16d1 2023-07-08 jrick strlen(markers[i])) == 0) {
224 3d1b16d1 2023-07-08 jrick if (strcmp(markers[i],
225 3d1b16d1 2023-07-08 jrick GOT_DIFF_CONFLICT_MARKER_END) == 0) {
226 3d1b16d1 2023-07-08 jrick *status = GOT_STATUS_CONFLICT;
227 3d1b16d1 2023-07-08 jrick goto done;
228 3d1b16d1 2023-07-08 jrick } else
229 3d1b16d1 2023-07-08 jrick i++;
230 3d1b16d1 2023-07-08 jrick }
231 3d1b16d1 2023-07-08 jrick ++ln;
232 3d1b16d1 2023-07-08 jrick }
233 3d1b16d1 2023-07-08 jrick }
234 3d1b16d1 2023-07-08 jrick
235 3d1b16d1 2023-07-08 jrick done:
236 3d1b16d1 2023-07-08 jrick free(line);
237 3d1b16d1 2023-07-08 jrick if (f1 != NULL && fclose(f1) == EOF && err == NULL)
238 3d1b16d1 2023-07-08 jrick err = got_error_from_errno("fclose");
239 3d1b16d1 2023-07-08 jrick free_err = got_diffreg_result_free(diffreg_result);
240 3d1b16d1 2023-07-08 jrick if (err == NULL)
241 3d1b16d1 2023-07-08 jrick err = free_err;
242 3d1b16d1 2023-07-08 jrick
243 3d1b16d1 2023-07-08 jrick return err;
244 3d1b16d1 2023-07-08 jrick }
245 3d1b16d1 2023-07-08 jrick
246 3d1b16d1 2023-07-08 jrick static int
247 3d1b16d1 2023-07-08 jrick xbit_differs(struct got_fileindex_entry *ie, uint16_t st_mode)
248 3d1b16d1 2023-07-08 jrick {
249 3d1b16d1 2023-07-08 jrick mode_t ie_mode = got_fileindex_perms_to_st(ie);
250 3d1b16d1 2023-07-08 jrick return ((ie_mode & S_IXUSR) != (st_mode & S_IXUSR));
251 3d1b16d1 2023-07-08 jrick }
252 3d1b16d1 2023-07-08 jrick
253 3d1b16d1 2023-07-08 jrick static int
254 3d1b16d1 2023-07-08 jrick stat_info_differs(struct got_fileindex_entry *ie, struct stat *sb)
255 3d1b16d1 2023-07-08 jrick {
256 3d1b16d1 2023-07-08 jrick return !(ie->ctime_sec == sb->st_ctim.tv_sec &&
257 3d1b16d1 2023-07-08 jrick ie->ctime_nsec == sb->st_ctim.tv_nsec &&
258 3d1b16d1 2023-07-08 jrick ie->mtime_sec == sb->st_mtim.tv_sec &&
259 3d1b16d1 2023-07-08 jrick ie->mtime_nsec == sb->st_mtim.tv_nsec &&
260 3d1b16d1 2023-07-08 jrick ie->size == (sb->st_size & 0xffffffff) &&
261 3d1b16d1 2023-07-08 jrick !xbit_differs(ie, sb->st_mode));
262 3d1b16d1 2023-07-08 jrick }
263 3d1b16d1 2023-07-08 jrick
264 3d1b16d1 2023-07-08 jrick static unsigned char
265 3d1b16d1 2023-07-08 jrick get_staged_status(struct got_fileindex_entry *ie)
266 3d1b16d1 2023-07-08 jrick {
267 3d1b16d1 2023-07-08 jrick switch (got_fileindex_entry_stage_get(ie)) {
268 3d1b16d1 2023-07-08 jrick case GOT_FILEIDX_STAGE_ADD:
269 3d1b16d1 2023-07-08 jrick return GOT_STATUS_ADD;
270 3d1b16d1 2023-07-08 jrick case GOT_FILEIDX_STAGE_DELETE:
271 3d1b16d1 2023-07-08 jrick return GOT_STATUS_DELETE;
272 3d1b16d1 2023-07-08 jrick case GOT_FILEIDX_STAGE_MODIFY:
273 3d1b16d1 2023-07-08 jrick return GOT_STATUS_MODIFY;
274 3d1b16d1 2023-07-08 jrick default:
275 3d1b16d1 2023-07-08 jrick return GOT_STATUS_NO_CHANGE;
276 3d1b16d1 2023-07-08 jrick }
277 3d1b16d1 2023-07-08 jrick }
278 3d1b16d1 2023-07-08 jrick
279 3d1b16d1 2023-07-08 jrick static const struct got_error *
280 3d1b16d1 2023-07-08 jrick get_symlink_modification_status(unsigned char *status,
281 3d1b16d1 2023-07-08 jrick struct got_fileindex_entry *ie, const char *abspath,
282 3d1b16d1 2023-07-08 jrick int dirfd, const char *de_name, struct got_blob_object *blob)
283 3d1b16d1 2023-07-08 jrick {
284 3d1b16d1 2023-07-08 jrick const struct got_error *err = NULL;
285 3d1b16d1 2023-07-08 jrick char target_path[PATH_MAX];
286 3d1b16d1 2023-07-08 jrick char etarget[PATH_MAX];
287 3d1b16d1 2023-07-08 jrick ssize_t elen;
288 3d1b16d1 2023-07-08 jrick size_t len, target_len = 0;
289 3d1b16d1 2023-07-08 jrick const uint8_t *buf = got_object_blob_get_read_buf(blob);
290 3d1b16d1 2023-07-08 jrick size_t hdrlen = got_object_blob_get_hdrlen(blob);
291 3d1b16d1 2023-07-08 jrick
292 3d1b16d1 2023-07-08 jrick *status = GOT_STATUS_NO_CHANGE;
293 3d1b16d1 2023-07-08 jrick
294 3d1b16d1 2023-07-08 jrick /* Blob object content specifies the target path of the link. */
295 3d1b16d1 2023-07-08 jrick do {
296 3d1b16d1 2023-07-08 jrick err = got_object_blob_read_block(&len, blob);
297 3d1b16d1 2023-07-08 jrick if (err)
298 3d1b16d1 2023-07-08 jrick return err;
299 3d1b16d1 2023-07-08 jrick if (len + target_len >= sizeof(target_path)) {
300 3d1b16d1 2023-07-08 jrick /*
301 3d1b16d1 2023-07-08 jrick * Should not happen. The blob contents were OK
302 3d1b16d1 2023-07-08 jrick * when this symlink was installed.
303 3d1b16d1 2023-07-08 jrick */
304 3d1b16d1 2023-07-08 jrick return got_error(GOT_ERR_NO_SPACE);
305 3d1b16d1 2023-07-08 jrick }
306 3d1b16d1 2023-07-08 jrick if (len > 0) {
307 3d1b16d1 2023-07-08 jrick /* Skip blob object header first time around. */
308 3d1b16d1 2023-07-08 jrick memcpy(target_path + target_len, buf + hdrlen,
309 3d1b16d1 2023-07-08 jrick len - hdrlen);
310 3d1b16d1 2023-07-08 jrick target_len += len - hdrlen;
311 3d1b16d1 2023-07-08 jrick hdrlen = 0;
312 3d1b16d1 2023-07-08 jrick }
313 3d1b16d1 2023-07-08 jrick } while (len != 0);
314 3d1b16d1 2023-07-08 jrick target_path[target_len] = '\0';
315 3d1b16d1 2023-07-08 jrick
316 3d1b16d1 2023-07-08 jrick if (dirfd != -1) {
317 3d1b16d1 2023-07-08 jrick elen = readlinkat(dirfd, de_name, etarget, sizeof(etarget));
318 3d1b16d1 2023-07-08 jrick if (elen == -1)
319 3d1b16d1 2023-07-08 jrick return got_error_from_errno2("readlinkat", abspath);
320 3d1b16d1 2023-07-08 jrick } else {
321 3d1b16d1 2023-07-08 jrick elen = readlink(abspath, etarget, sizeof(etarget));
322 3d1b16d1 2023-07-08 jrick if (elen == -1)
323 3d1b16d1 2023-07-08 jrick return got_error_from_errno2("readlink", abspath);
324 3d1b16d1 2023-07-08 jrick }
325 3d1b16d1 2023-07-08 jrick
326 3d1b16d1 2023-07-08 jrick if (elen != target_len || memcmp(etarget, target_path, target_len) != 0)
327 3d1b16d1 2023-07-08 jrick *status = GOT_STATUS_MODIFY;
328 3d1b16d1 2023-07-08 jrick
329 3d1b16d1 2023-07-08 jrick return NULL;
330 3d1b16d1 2023-07-08 jrick }
331 3d1b16d1 2023-07-08 jrick
332 3d1b16d1 2023-07-08 jrick static const struct got_error *
333 3d1b16d1 2023-07-08 jrick get_file_status(unsigned char *status, struct stat *sb,
334 3d1b16d1 2023-07-08 jrick struct got_fileindex_entry *ie, const char *abspath,
335 3d1b16d1 2023-07-08 jrick int dirfd, const char *de_name, struct got_repository *repo)
336 3d1b16d1 2023-07-08 jrick {
337 3d1b16d1 2023-07-08 jrick const struct got_error *err = NULL;
338 3d1b16d1 2023-07-08 jrick struct got_object_id id;
339 3d1b16d1 2023-07-08 jrick size_t hdrlen;
340 3d1b16d1 2023-07-08 jrick int fd = -1, fd1 = -1;
341 3d1b16d1 2023-07-08 jrick FILE *f = NULL;
342 3d1b16d1 2023-07-08 jrick uint8_t fbuf[8192];
343 3d1b16d1 2023-07-08 jrick struct got_blob_object *blob = NULL;
344 3d1b16d1 2023-07-08 jrick size_t flen, blen;
345 3d1b16d1 2023-07-08 jrick unsigned char staged_status;
346 3d1b16d1 2023-07-08 jrick
347 3d1b16d1 2023-07-08 jrick staged_status = get_staged_status(ie);
348 3d1b16d1 2023-07-08 jrick *status = GOT_STATUS_NO_CHANGE;
349 3d1b16d1 2023-07-08 jrick memset(sb, 0, sizeof(*sb));
350 3d1b16d1 2023-07-08 jrick
351 3d1b16d1 2023-07-08 jrick /*
352 3d1b16d1 2023-07-08 jrick * Whenever the caller provides a directory descriptor and a
353 3d1b16d1 2023-07-08 jrick * directory entry name for the file, use them! This prevents
354 3d1b16d1 2023-07-08 jrick * race conditions if filesystem paths change beneath our feet.
355 3d1b16d1 2023-07-08 jrick */
356 3d1b16d1 2023-07-08 jrick if (dirfd != -1) {
357 3d1b16d1 2023-07-08 jrick if (fstatat(dirfd, de_name, sb, AT_SYMLINK_NOFOLLOW) == -1) {
358 3d1b16d1 2023-07-08 jrick if (errno == ENOENT) {
359 3d1b16d1 2023-07-08 jrick if (got_fileindex_entry_has_file_on_disk(ie))
360 3d1b16d1 2023-07-08 jrick *status = GOT_STATUS_MISSING;
361 3d1b16d1 2023-07-08 jrick else
362 3d1b16d1 2023-07-08 jrick *status = GOT_STATUS_DELETE;
363 3d1b16d1 2023-07-08 jrick goto done;
364 3d1b16d1 2023-07-08 jrick }
365 3d1b16d1 2023-07-08 jrick err = got_error_from_errno2("fstatat", abspath);
366 3d1b16d1 2023-07-08 jrick goto done;
367 3d1b16d1 2023-07-08 jrick }
368 3d1b16d1 2023-07-08 jrick } else {
369 3d1b16d1 2023-07-08 jrick fd = open(abspath, O_RDONLY | O_NOFOLLOW | O_CLOEXEC);
370 3d1b16d1 2023-07-08 jrick if (fd == -1 && errno != ENOENT &&
371 3d1b16d1 2023-07-08 jrick !got_err_open_nofollow_on_symlink())
372 3d1b16d1 2023-07-08 jrick return got_error_from_errno2("open", abspath);
373 3d1b16d1 2023-07-08 jrick else if (fd == -1 && got_err_open_nofollow_on_symlink()) {
374 3d1b16d1 2023-07-08 jrick if (lstat(abspath, sb) == -1)
375 3d1b16d1 2023-07-08 jrick return got_error_from_errno2("lstat", abspath);
376 3d1b16d1 2023-07-08 jrick } else if (fd == -1 || fstat(fd, sb) == -1) {
377 3d1b16d1 2023-07-08 jrick if (errno == ENOENT) {
378 3d1b16d1 2023-07-08 jrick if (got_fileindex_entry_has_file_on_disk(ie))
379 3d1b16d1 2023-07-08 jrick *status = GOT_STATUS_MISSING;
380 3d1b16d1 2023-07-08 jrick else
381 3d1b16d1 2023-07-08 jrick *status = GOT_STATUS_DELETE;
382 3d1b16d1 2023-07-08 jrick goto done;
383 3d1b16d1 2023-07-08 jrick }
384 3d1b16d1 2023-07-08 jrick err = got_error_from_errno2("fstat", abspath);
385 3d1b16d1 2023-07-08 jrick goto done;
386 3d1b16d1 2023-07-08 jrick }
387 3d1b16d1 2023-07-08 jrick }
388 3d1b16d1 2023-07-08 jrick
389 3d1b16d1 2023-07-08 jrick if (!S_ISREG(sb->st_mode) && !S_ISLNK(sb->st_mode)) {
390 3d1b16d1 2023-07-08 jrick *status = GOT_STATUS_OBSTRUCTED;
391 3d1b16d1 2023-07-08 jrick goto done;
392 3d1b16d1 2023-07-08 jrick }
393 3d1b16d1 2023-07-08 jrick
394 3d1b16d1 2023-07-08 jrick if (!got_fileindex_entry_has_file_on_disk(ie)) {
395 3d1b16d1 2023-07-08 jrick *status = GOT_STATUS_DELETE;
396 3d1b16d1 2023-07-08 jrick goto done;
397 3d1b16d1 2023-07-08 jrick } else if (!got_fileindex_entry_has_blob(ie) &&
398 3d1b16d1 2023-07-08 jrick staged_status != GOT_STATUS_ADD) {
399 3d1b16d1 2023-07-08 jrick *status = GOT_STATUS_ADD;
400 3d1b16d1 2023-07-08 jrick goto done;
401 3d1b16d1 2023-07-08 jrick }
402 3d1b16d1 2023-07-08 jrick
403 3d1b16d1 2023-07-08 jrick if (!stat_info_differs(ie, sb))
404 3d1b16d1 2023-07-08 jrick goto done;
405 3d1b16d1 2023-07-08 jrick
406 3d1b16d1 2023-07-08 jrick if (S_ISLNK(sb->st_mode) &&
407 3d1b16d1 2023-07-08 jrick got_fileindex_entry_filetype_get(ie) != GOT_FILEIDX_MODE_SYMLINK) {
408 3d1b16d1 2023-07-08 jrick *status = GOT_STATUS_MODIFY;
409 3d1b16d1 2023-07-08 jrick goto done;
410 3d1b16d1 2023-07-08 jrick }
411 3d1b16d1 2023-07-08 jrick
412 3d1b16d1 2023-07-08 jrick if (staged_status == GOT_STATUS_MODIFY ||
413 3d1b16d1 2023-07-08 jrick staged_status == GOT_STATUS_ADD)
414 3d1b16d1 2023-07-08 jrick got_fileindex_entry_get_staged_blob_id(&id, ie);
415 3d1b16d1 2023-07-08 jrick else
416 3d1b16d1 2023-07-08 jrick got_fileindex_entry_get_blob_id(&id, ie);
417 3d1b16d1 2023-07-08 jrick
418 3d1b16d1 2023-07-08 jrick fd1 = got_opentempfd();
419 3d1b16d1 2023-07-08 jrick if (fd1 == -1) {
420 3d1b16d1 2023-07-08 jrick err = got_error_from_errno("got_opentempfd");
421 3d1b16d1 2023-07-08 jrick goto done;
422 3d1b16d1 2023-07-08 jrick }
423 3d1b16d1 2023-07-08 jrick err = got_object_open_as_blob(&blob, repo, &id, sizeof(fbuf), fd1);
424 3d1b16d1 2023-07-08 jrick if (err)
425 3d1b16d1 2023-07-08 jrick goto done;
426 3d1b16d1 2023-07-08 jrick
427 3d1b16d1 2023-07-08 jrick if (S_ISLNK(sb->st_mode)) {
428 3d1b16d1 2023-07-08 jrick err = get_symlink_modification_status(status, ie,
429 3d1b16d1 2023-07-08 jrick abspath, dirfd, de_name, blob);
430 3d1b16d1 2023-07-08 jrick goto done;
431 3d1b16d1 2023-07-08 jrick }
432 3d1b16d1 2023-07-08 jrick
433 3d1b16d1 2023-07-08 jrick if (dirfd != -1) {
434 3d1b16d1 2023-07-08 jrick fd = openat(dirfd, de_name, O_RDONLY | O_NOFOLLOW | O_CLOEXEC);
435 3d1b16d1 2023-07-08 jrick if (fd == -1) {
436 3d1b16d1 2023-07-08 jrick err = got_error_from_errno2("openat", abspath);
437 3d1b16d1 2023-07-08 jrick goto done;
438 3d1b16d1 2023-07-08 jrick }
439 3d1b16d1 2023-07-08 jrick }
440 3d1b16d1 2023-07-08 jrick
441 3d1b16d1 2023-07-08 jrick f = fdopen(fd, "r");
442 3d1b16d1 2023-07-08 jrick if (f == NULL) {
443 3d1b16d1 2023-07-08 jrick err = got_error_from_errno2("fdopen", abspath);
444 3d1b16d1 2023-07-08 jrick goto done;
445 3d1b16d1 2023-07-08 jrick }
446 3d1b16d1 2023-07-08 jrick fd = -1;
447 3d1b16d1 2023-07-08 jrick hdrlen = got_object_blob_get_hdrlen(blob);
448 3d1b16d1 2023-07-08 jrick for (;;) {
449 3d1b16d1 2023-07-08 jrick const uint8_t *bbuf = got_object_blob_get_read_buf(blob);
450 3d1b16d1 2023-07-08 jrick err = got_object_blob_read_block(&blen, blob);
451 3d1b16d1 2023-07-08 jrick if (err)
452 3d1b16d1 2023-07-08 jrick goto done;
453 3d1b16d1 2023-07-08 jrick /* Skip length of blob object header first time around. */
454 3d1b16d1 2023-07-08 jrick flen = fread(fbuf, 1, sizeof(fbuf) - hdrlen, f);
455 3d1b16d1 2023-07-08 jrick if (flen == 0 && ferror(f)) {
456 3d1b16d1 2023-07-08 jrick err = got_error_from_errno("fread");
457 3d1b16d1 2023-07-08 jrick goto done;
458 3d1b16d1 2023-07-08 jrick }
459 3d1b16d1 2023-07-08 jrick if (blen - hdrlen == 0) {
460 3d1b16d1 2023-07-08 jrick if (flen != 0)
461 3d1b16d1 2023-07-08 jrick *status = GOT_STATUS_MODIFY;
462 3d1b16d1 2023-07-08 jrick break;
463 3d1b16d1 2023-07-08 jrick } else if (flen == 0) {
464 3d1b16d1 2023-07-08 jrick if (blen - hdrlen != 0)
465 3d1b16d1 2023-07-08 jrick *status = GOT_STATUS_MODIFY;
466 3d1b16d1 2023-07-08 jrick break;
467 3d1b16d1 2023-07-08 jrick } else if (blen - hdrlen == flen) {
468 3d1b16d1 2023-07-08 jrick /* Skip blob object header first time around. */
469 3d1b16d1 2023-07-08 jrick if (memcmp(bbuf + hdrlen, fbuf, flen) != 0) {
470 3d1b16d1 2023-07-08 jrick *status = GOT_STATUS_MODIFY;
471 3d1b16d1 2023-07-08 jrick break;
472 3d1b16d1 2023-07-08 jrick }
473 3d1b16d1 2023-07-08 jrick } else {
474 3d1b16d1 2023-07-08 jrick *status = GOT_STATUS_MODIFY;
475 3d1b16d1 2023-07-08 jrick break;
476 3d1b16d1 2023-07-08 jrick }
477 3d1b16d1 2023-07-08 jrick hdrlen = 0;
478 3d1b16d1 2023-07-08 jrick }
479 3d1b16d1 2023-07-08 jrick
480 3d1b16d1 2023-07-08 jrick if (*status == GOT_STATUS_MODIFY) {
481 3d1b16d1 2023-07-08 jrick rewind(f);
482 3d1b16d1 2023-07-08 jrick err = get_modified_file_content_status(status, blob, ie->path,
483 3d1b16d1 2023-07-08 jrick sb, f);
484 3d1b16d1 2023-07-08 jrick } else if (xbit_differs(ie, sb->st_mode))
485 3d1b16d1 2023-07-08 jrick *status = GOT_STATUS_MODE_CHANGE;
486 3d1b16d1 2023-07-08 jrick done:
487 3d1b16d1 2023-07-08 jrick if (fd1 != -1 && close(fd1) == -1 && err == NULL)
488 3d1b16d1 2023-07-08 jrick err = got_error_from_errno("close");
489 3d1b16d1 2023-07-08 jrick if (blob)
490 3d1b16d1 2023-07-08 jrick got_object_blob_close(blob);
491 3d1b16d1 2023-07-08 jrick if (f != NULL && fclose(f) == EOF && err == NULL)
492 3d1b16d1 2023-07-08 jrick err = got_error_from_errno2("fclose", abspath);
493 3d1b16d1 2023-07-08 jrick if (fd != -1 && close(fd) == -1 && err == NULL)
494 3d1b16d1 2023-07-08 jrick err = got_error_from_errno2("close", abspath);
495 3d1b16d1 2023-07-08 jrick return err;
496 3d1b16d1 2023-07-08 jrick }
497 3d1b16d1 2023-07-08 jrick
498 3d1b16d1 2023-07-08 jrick static const struct got_error *
499 3d1b16d1 2023-07-08 jrick get_ref_name(char **refname, struct got_worktree *worktree, const char *prefix)
500 3d1b16d1 2023-07-08 jrick {
501 3d1b16d1 2023-07-08 jrick const struct got_error *err = NULL;
502 3d1b16d1 2023-07-08 jrick char *uuidstr = NULL;
503 3d1b16d1 2023-07-08 jrick
504 3d1b16d1 2023-07-08 jrick *refname = NULL;
505 3d1b16d1 2023-07-08 jrick
506 3d1b16d1 2023-07-08 jrick err = got_worktree_get_uuid(&uuidstr, worktree);
507 3d1b16d1 2023-07-08 jrick if (err)
508 3d1b16d1 2023-07-08 jrick return err;
509 3d1b16d1 2023-07-08 jrick
510 3d1b16d1 2023-07-08 jrick if (asprintf(refname, "%s-%s", prefix, uuidstr) == -1) {
511 3d1b16d1 2023-07-08 jrick err = got_error_from_errno("asprintf");
512 3d1b16d1 2023-07-08 jrick *refname = NULL;
513 3d1b16d1 2023-07-08 jrick }
514 3d1b16d1 2023-07-08 jrick free(uuidstr);
515 3d1b16d1 2023-07-08 jrick return err;
516 3d1b16d1 2023-07-08 jrick }
517 3d1b16d1 2023-07-08 jrick
518 3d1b16d1 2023-07-08 jrick static const struct got_error *
519 3d1b16d1 2023-07-08 jrick get_base_ref_name(char **refname, struct got_worktree *worktree)
520 3d1b16d1 2023-07-08 jrick {
521 3d1b16d1 2023-07-08 jrick return get_ref_name(refname, worktree, GOT_WORKTREE_BASE_REF_PREFIX);
522 3d1b16d1 2023-07-08 jrick }
523 3d1b16d1 2023-07-08 jrick
524 3d1b16d1 2023-07-08 jrick /*
525 3d1b16d1 2023-07-08 jrick * Prevent Git's garbage collector from deleting our base commit by
526 3d1b16d1 2023-07-08 jrick * setting a reference to our base commit's ID.
527 3d1b16d1 2023-07-08 jrick */
528 3d1b16d1 2023-07-08 jrick static const struct got_error *
529 3d1b16d1 2023-07-08 jrick ref_base_commit(struct got_worktree *worktree, struct got_repository *repo)
530 3d1b16d1 2023-07-08 jrick {
531 3d1b16d1 2023-07-08 jrick const struct got_error *err = NULL;
532 3d1b16d1 2023-07-08 jrick struct got_reference *ref = NULL;
533 3d1b16d1 2023-07-08 jrick char *refname;
534 3d1b16d1 2023-07-08 jrick
535 3d1b16d1 2023-07-08 jrick err = get_base_ref_name(&refname, worktree);
536 3d1b16d1 2023-07-08 jrick if (err)
537 3d1b16d1 2023-07-08 jrick return err;
538 3d1b16d1 2023-07-08 jrick
539 3d1b16d1 2023-07-08 jrick err = got_ref_alloc(&ref, refname, worktree->base_commit_id);
540 3d1b16d1 2023-07-08 jrick if (err)
541 3d1b16d1 2023-07-08 jrick goto done;
542 3d1b16d1 2023-07-08 jrick
543 3d1b16d1 2023-07-08 jrick err = got_ref_write(ref, repo);
544 3d1b16d1 2023-07-08 jrick done:
545 3d1b16d1 2023-07-08 jrick free(refname);
546 3d1b16d1 2023-07-08 jrick if (ref)
547 3d1b16d1 2023-07-08 jrick got_ref_close(ref);
548 3d1b16d1 2023-07-08 jrick return err;
549 3d1b16d1 2023-07-08 jrick }
550 3d1b16d1 2023-07-08 jrick
551 3d1b16d1 2023-07-08 jrick static const struct got_error *
552 3d1b16d1 2023-07-08 jrick get_fileindex_path(char **fileindex_path, struct got_worktree *worktree)
553 3d1b16d1 2023-07-08 jrick {
554 3d1b16d1 2023-07-08 jrick const struct got_error *err = NULL;
555 3d1b16d1 2023-07-08 jrick
556 3d1b16d1 2023-07-08 jrick if (asprintf(fileindex_path, "%s/%s/%s", worktree->root_path,
557 3d1b16d1 2023-07-08 jrick GOT_WORKTREE_GOT_DIR, GOT_WORKTREE_FILE_INDEX) == -1) {
558 3d1b16d1 2023-07-08 jrick err = got_error_from_errno("asprintf");
559 3d1b16d1 2023-07-08 jrick *fileindex_path = NULL;
560 3d1b16d1 2023-07-08 jrick }
561 3d1b16d1 2023-07-08 jrick return err;
562 3d1b16d1 2023-07-08 jrick }
563 3d1b16d1 2023-07-08 jrick
564 3d1b16d1 2023-07-08 jrick static const struct got_error *
565 3d1b16d1 2023-07-08 jrick open_fileindex(struct got_fileindex **fileindex, char **fileindex_path,
566 3d1b16d1 2023-07-08 jrick struct got_worktree *worktree)
567 3d1b16d1 2023-07-08 jrick {
568 3d1b16d1 2023-07-08 jrick const struct got_error *err = NULL;
569 3d1b16d1 2023-07-08 jrick FILE *index = NULL;
570 3d1b16d1 2023-07-08 jrick
571 3d1b16d1 2023-07-08 jrick *fileindex_path = NULL;
572 3d1b16d1 2023-07-08 jrick *fileindex = got_fileindex_alloc();
573 3d1b16d1 2023-07-08 jrick if (*fileindex == NULL)
574 3d1b16d1 2023-07-08 jrick return got_error_from_errno("got_fileindex_alloc");
575 3d1b16d1 2023-07-08 jrick
576 3d1b16d1 2023-07-08 jrick err = get_fileindex_path(fileindex_path, worktree);
577 3d1b16d1 2023-07-08 jrick if (err)
578 3d1b16d1 2023-07-08 jrick goto done;
579 3d1b16d1 2023-07-08 jrick
580 3d1b16d1 2023-07-08 jrick index = fopen(*fileindex_path, "rbe");
581 3d1b16d1 2023-07-08 jrick if (index == NULL) {
582 3d1b16d1 2023-07-08 jrick if (errno != ENOENT)
583 3d1b16d1 2023-07-08 jrick err = got_error_from_errno2("fopen", *fileindex_path);
584 3d1b16d1 2023-07-08 jrick } else {
585 3d1b16d1 2023-07-08 jrick err = got_fileindex_read(*fileindex, index);
586 3d1b16d1 2023-07-08 jrick if (fclose(index) == EOF && err == NULL)
587 3d1b16d1 2023-07-08 jrick err = got_error_from_errno("fclose");
588 3d1b16d1 2023-07-08 jrick }
589 3d1b16d1 2023-07-08 jrick done:
590 3d1b16d1 2023-07-08 jrick if (err) {
591 3d1b16d1 2023-07-08 jrick free(*fileindex_path);
592 3d1b16d1 2023-07-08 jrick *fileindex_path = NULL;
593 3d1b16d1 2023-07-08 jrick got_fileindex_free(*fileindex);
594 3d1b16d1 2023-07-08 jrick *fileindex = NULL;
595 3d1b16d1 2023-07-08 jrick }
596 3d1b16d1 2023-07-08 jrick return err;
597 3d1b16d1 2023-07-08 jrick }
598 3d1b16d1 2023-07-08 jrick
599 3d1b16d1 2023-07-08 jrick static const struct got_error *
600 3d1b16d1 2023-07-08 jrick sync_fileindex(struct got_fileindex *fileindex, const char *fileindex_path)
601 3d1b16d1 2023-07-08 jrick {
602 3d1b16d1 2023-07-08 jrick const struct got_error *err = NULL;
603 3d1b16d1 2023-07-08 jrick char *new_fileindex_path = NULL;
604 3d1b16d1 2023-07-08 jrick FILE *new_index = NULL;
605 3d1b16d1 2023-07-08 jrick struct timespec timeout;
606 3d1b16d1 2023-07-08 jrick
607 3d1b16d1 2023-07-08 jrick err = got_opentemp_named(&new_fileindex_path, &new_index,
608 3d1b16d1 2023-07-08 jrick fileindex_path, "");
609 3d1b16d1 2023-07-08 jrick if (err)
610 3d1b16d1 2023-07-08 jrick goto done;
611 3d1b16d1 2023-07-08 jrick
612 3d1b16d1 2023-07-08 jrick err = got_fileindex_write(fileindex, new_index);
613 3d1b16d1 2023-07-08 jrick if (err)
614 3d1b16d1 2023-07-08 jrick goto done;
615 3d1b16d1 2023-07-08 jrick
616 3d1b16d1 2023-07-08 jrick if (rename(new_fileindex_path, fileindex_path) != 0) {
617 3d1b16d1 2023-07-08 jrick err = got_error_from_errno3("rename", new_fileindex_path,
618 3d1b16d1 2023-07-08 jrick fileindex_path);
619 3d1b16d1 2023-07-08 jrick unlink(new_fileindex_path);
620 3d1b16d1 2023-07-08 jrick }
621 3d1b16d1 2023-07-08 jrick
622 3d1b16d1 2023-07-08 jrick /*
623 3d1b16d1 2023-07-08 jrick * Sleep for a short amount of time to ensure that files modified after
624 3d1b16d1 2023-07-08 jrick * this program exits have a different time stamp from the one which
625 3d1b16d1 2023-07-08 jrick * was recorded in the file index.
626 3d1b16d1 2023-07-08 jrick */
627 3d1b16d1 2023-07-08 jrick timeout.tv_sec = 0;
628 3d1b16d1 2023-07-08 jrick timeout.tv_nsec = 1;
629 3d1b16d1 2023-07-08 jrick nanosleep(&timeout, NULL);
630 3d1b16d1 2023-07-08 jrick done:
631 3d1b16d1 2023-07-08 jrick if (new_index)
632 3d1b16d1 2023-07-08 jrick fclose(new_index);
633 3d1b16d1 2023-07-08 jrick free(new_fileindex_path);
634 3d1b16d1 2023-07-08 jrick return err;
635 3d1b16d1 2023-07-08 jrick }
636 3d1b16d1 2023-07-08 jrick
637 3d1b16d1 2023-07-08 jrick struct diff_dir_cb_arg {
638 3d1b16d1 2023-07-08 jrick struct got_fileindex *fileindex;
639 3d1b16d1 2023-07-08 jrick struct got_worktree *worktree;
640 3d1b16d1 2023-07-08 jrick const char *status_path;
641 3d1b16d1 2023-07-08 jrick size_t status_path_len;
642 3d1b16d1 2023-07-08 jrick struct got_repository *repo;
643 3d1b16d1 2023-07-08 jrick got_worktree_status_cb status_cb;
644 3d1b16d1 2023-07-08 jrick void *status_arg;
645 3d1b16d1 2023-07-08 jrick got_cancel_cb cancel_cb;
646 3d1b16d1 2023-07-08 jrick void *cancel_arg;
647 3d1b16d1 2023-07-08 jrick /* A pathlist containing per-directory pathlists of ignore patterns. */
648 3d1b16d1 2023-07-08 jrick struct got_pathlist_head *ignores;
649 3d1b16d1 2023-07-08 jrick int report_unchanged;
650 3d1b16d1 2023-07-08 jrick int no_ignores;
651 3d1b16d1 2023-07-08 jrick };
652 3d1b16d1 2023-07-08 jrick
653 3d1b16d1 2023-07-08 jrick static const struct got_error *
654 3d1b16d1 2023-07-08 jrick report_file_status(struct got_fileindex_entry *ie, const char *abspath,
655 3d1b16d1 2023-07-08 jrick int dirfd, const char *de_name,
656 3d1b16d1 2023-07-08 jrick got_worktree_status_cb status_cb, void *status_arg,
657 3d1b16d1 2023-07-08 jrick struct got_repository *repo, int report_unchanged)
658 3d1b16d1 2023-07-08 jrick {
659 3d1b16d1 2023-07-08 jrick const struct got_error *err = NULL;
660 3d1b16d1 2023-07-08 jrick unsigned char status = GOT_STATUS_NO_CHANGE;
661 3d1b16d1 2023-07-08 jrick unsigned char staged_status;
662 3d1b16d1 2023-07-08 jrick struct stat sb;
663 3d1b16d1 2023-07-08 jrick struct got_object_id blob_id, commit_id, staged_blob_id;
664 3d1b16d1 2023-07-08 jrick struct got_object_id *blob_idp = NULL, *commit_idp = NULL;
665 3d1b16d1 2023-07-08 jrick struct got_object_id *staged_blob_idp = NULL;
666 3d1b16d1 2023-07-08 jrick
667 3d1b16d1 2023-07-08 jrick staged_status = get_staged_status(ie);
668 3d1b16d1 2023-07-08 jrick err = get_file_status(&status, &sb, ie, abspath, dirfd, de_name, repo);
669 3d1b16d1 2023-07-08 jrick if (err)
670 3d1b16d1 2023-07-08 jrick return err;
671 3d1b16d1 2023-07-08 jrick
672 3d1b16d1 2023-07-08 jrick if (status == GOT_STATUS_NO_CHANGE &&
673 3d1b16d1 2023-07-08 jrick staged_status == GOT_STATUS_NO_CHANGE && !report_unchanged)
674 3d1b16d1 2023-07-08 jrick return NULL;
675 3d1b16d1 2023-07-08 jrick
676 3d1b16d1 2023-07-08 jrick if (got_fileindex_entry_has_blob(ie))
677 3d1b16d1 2023-07-08 jrick blob_idp = got_fileindex_entry_get_blob_id(&blob_id, ie);
678 3d1b16d1 2023-07-08 jrick if (got_fileindex_entry_has_commit(ie))
679 3d1b16d1 2023-07-08 jrick commit_idp = got_fileindex_entry_get_commit_id(&commit_id, ie);
680 3d1b16d1 2023-07-08 jrick if (staged_status == GOT_STATUS_ADD ||
681 3d1b16d1 2023-07-08 jrick staged_status == GOT_STATUS_MODIFY) {
682 3d1b16d1 2023-07-08 jrick staged_blob_idp = got_fileindex_entry_get_staged_blob_id(
683 3d1b16d1 2023-07-08 jrick &staged_blob_id, ie);
684 3d1b16d1 2023-07-08 jrick }
685 3d1b16d1 2023-07-08 jrick
686 3d1b16d1 2023-07-08 jrick return (*status_cb)(status_arg, status, staged_status,
687 3d1b16d1 2023-07-08 jrick ie->path, blob_idp, staged_blob_idp, commit_idp, dirfd, de_name);
688 3d1b16d1 2023-07-08 jrick }
689 3d1b16d1 2023-07-08 jrick
690 3d1b16d1 2023-07-08 jrick static const struct got_error *
691 3d1b16d1 2023-07-08 jrick status_old_new(void *arg, struct got_fileindex_entry *ie,
692 3d1b16d1 2023-07-08 jrick struct dirent *de, const char *parent_path, int dirfd)
693 3d1b16d1 2023-07-08 jrick {
694 3d1b16d1 2023-07-08 jrick const struct got_error *err = NULL;
695 3d1b16d1 2023-07-08 jrick struct diff_dir_cb_arg *a = arg;
696 3d1b16d1 2023-07-08 jrick char *abspath;
697 3d1b16d1 2023-07-08 jrick
698 f6b8c3c2 2023-07-24 stsp if (a->cancel_cb) {
699 f6b8c3c2 2023-07-24 stsp err = a->cancel_cb(a->cancel_arg);
700 f6b8c3c2 2023-07-24 stsp if (err)
701 f6b8c3c2 2023-07-24 stsp return err;
702 f6b8c3c2 2023-07-24 stsp }
703 3d1b16d1 2023-07-08 jrick
704 3d1b16d1 2023-07-08 jrick if (got_path_cmp(parent_path, a->status_path,
705 3d1b16d1 2023-07-08 jrick strlen(parent_path), a->status_path_len) != 0 &&
706 3d1b16d1 2023-07-08 jrick !got_path_is_child(parent_path, a->status_path, a->status_path_len))
707 3d1b16d1 2023-07-08 jrick return NULL;
708 3d1b16d1 2023-07-08 jrick
709 3d1b16d1 2023-07-08 jrick if (parent_path[0]) {
710 3d1b16d1 2023-07-08 jrick if (asprintf(&abspath, "%s/%s/%s", a->worktree->root_path,
711 3d1b16d1 2023-07-08 jrick parent_path, de->d_name) == -1)
712 3d1b16d1 2023-07-08 jrick return got_error_from_errno("asprintf");
713 3d1b16d1 2023-07-08 jrick } else {
714 3d1b16d1 2023-07-08 jrick if (asprintf(&abspath, "%s/%s", a->worktree->root_path,
715 3d1b16d1 2023-07-08 jrick de->d_name) == -1)
716 3d1b16d1 2023-07-08 jrick return got_error_from_errno("asprintf");
717 3d1b16d1 2023-07-08 jrick }
718 3d1b16d1 2023-07-08 jrick
719 3d1b16d1 2023-07-08 jrick err = report_file_status(ie, abspath, dirfd, de->d_name,
720 3d1b16d1 2023-07-08 jrick a->status_cb, a->status_arg, a->repo, a->report_unchanged);
721 3d1b16d1 2023-07-08 jrick free(abspath);
722 3d1b16d1 2023-07-08 jrick return err;
723 3d1b16d1 2023-07-08 jrick }
724 3d1b16d1 2023-07-08 jrick
725 3d1b16d1 2023-07-08 jrick static const struct got_error *
726 3d1b16d1 2023-07-08 jrick status_old(void *arg, struct got_fileindex_entry *ie, const char *parent_path)
727 3d1b16d1 2023-07-08 jrick {
728 f6b8c3c2 2023-07-24 stsp const struct got_error *err = NULL;
729 3d1b16d1 2023-07-08 jrick struct diff_dir_cb_arg *a = arg;
730 3d1b16d1 2023-07-08 jrick struct got_object_id blob_id, commit_id;
731 3d1b16d1 2023-07-08 jrick unsigned char status;
732 3d1b16d1 2023-07-08 jrick
733 f6b8c3c2 2023-07-24 stsp if (a->cancel_cb) {
734 f6b8c3c2 2023-07-24 stsp err = a->cancel_cb(a->cancel_arg);
735 f6b8c3c2 2023-07-24 stsp if (err)
736 f6b8c3c2 2023-07-24 stsp return err;
737 f6b8c3c2 2023-07-24 stsp }
738 3d1b16d1 2023-07-08 jrick
739 3d1b16d1 2023-07-08 jrick if (!got_path_is_child(ie->path, a->status_path, a->status_path_len))
740 3d1b16d1 2023-07-08 jrick return NULL;
741 3d1b16d1 2023-07-08 jrick
742 3d1b16d1 2023-07-08 jrick got_fileindex_entry_get_blob_id(&blob_id, ie);
743 3d1b16d1 2023-07-08 jrick got_fileindex_entry_get_commit_id(&commit_id, ie);
744 3d1b16d1 2023-07-08 jrick if (got_fileindex_entry_has_file_on_disk(ie))
745 3d1b16d1 2023-07-08 jrick status = GOT_STATUS_MISSING;
746 3d1b16d1 2023-07-08 jrick else
747 3d1b16d1 2023-07-08 jrick status = GOT_STATUS_DELETE;
748 3d1b16d1 2023-07-08 jrick return (*a->status_cb)(a->status_arg, status, get_staged_status(ie),
749 3d1b16d1 2023-07-08 jrick ie->path, &blob_id, NULL, &commit_id, -1, NULL);
750 3d1b16d1 2023-07-08 jrick }
751 3d1b16d1 2023-07-08 jrick
752 3d1b16d1 2023-07-08 jrick static void
753 3d1b16d1 2023-07-08 jrick free_ignores(struct got_pathlist_head *ignores)
754 3d1b16d1 2023-07-08 jrick {
755 3d1b16d1 2023-07-08 jrick struct got_pathlist_entry *pe;
756 3d1b16d1 2023-07-08 jrick
757 3d1b16d1 2023-07-08 jrick TAILQ_FOREACH(pe, ignores, entry) {
758 3d1b16d1 2023-07-08 jrick struct got_pathlist_head *ignorelist = pe->data;
759 3d1b16d1 2023-07-08 jrick
760 3d1b16d1 2023-07-08 jrick got_pathlist_free(ignorelist, GOT_PATHLIST_FREE_PATH);
761 3d1b16d1 2023-07-08 jrick }
762 3d1b16d1 2023-07-08 jrick got_pathlist_free(ignores, GOT_PATHLIST_FREE_PATH);
763 3d1b16d1 2023-07-08 jrick }
764 3d1b16d1 2023-07-08 jrick
765 3d1b16d1 2023-07-08 jrick static const struct got_error *
766 3d1b16d1 2023-07-08 jrick read_ignores(struct got_pathlist_head *ignores, const char *path, FILE *f)
767 3d1b16d1 2023-07-08 jrick {
768 3d1b16d1 2023-07-08 jrick const struct got_error *err = NULL;
769 3d1b16d1 2023-07-08 jrick struct got_pathlist_entry *pe = NULL;
770 3d1b16d1 2023-07-08 jrick struct got_pathlist_head *ignorelist;
771 3d1b16d1 2023-07-08 jrick char *line = NULL, *pattern, *dirpath = NULL;
772 3d1b16d1 2023-07-08 jrick size_t linesize = 0;
773 3d1b16d1 2023-07-08 jrick ssize_t linelen;
774 3d1b16d1 2023-07-08 jrick
775 3d1b16d1 2023-07-08 jrick ignorelist = calloc(1, sizeof(*ignorelist));
776 3d1b16d1 2023-07-08 jrick if (ignorelist == NULL)
777 3d1b16d1 2023-07-08 jrick return got_error_from_errno("calloc");
778 3d1b16d1 2023-07-08 jrick TAILQ_INIT(ignorelist);
779 3d1b16d1 2023-07-08 jrick
780 3d1b16d1 2023-07-08 jrick while ((linelen = getline(&line, &linesize, f)) != -1) {
781 3d1b16d1 2023-07-08 jrick if (linelen > 0 && line[linelen - 1] == '\n')
782 3d1b16d1 2023-07-08 jrick line[linelen - 1] = '\0';
783 3d1b16d1 2023-07-08 jrick
784 3d1b16d1 2023-07-08 jrick /* Git's ignores may contain comments. */
785 3d1b16d1 2023-07-08 jrick if (line[0] == '#')
786 3d1b16d1 2023-07-08 jrick continue;
787 3d1b16d1 2023-07-08 jrick
788 3d1b16d1 2023-07-08 jrick /* Git's negated patterns are not (yet?) supported. */
789 3d1b16d1 2023-07-08 jrick if (line[0] == '!')
790 3d1b16d1 2023-07-08 jrick continue;
791 3d1b16d1 2023-07-08 jrick
792 3d1b16d1 2023-07-08 jrick if (asprintf(&pattern, "%s%s%s", path, path[0] ? "/" : "",
793 3d1b16d1 2023-07-08 jrick line) == -1) {
794 3d1b16d1 2023-07-08 jrick err = got_error_from_errno("asprintf");
795 3d1b16d1 2023-07-08 jrick goto done;
796 3d1b16d1 2023-07-08 jrick }
797 3d1b16d1 2023-07-08 jrick err = got_pathlist_insert(NULL, ignorelist, pattern, NULL);
798 3d1b16d1 2023-07-08 jrick if (err)
799 3d1b16d1 2023-07-08 jrick goto done;
800 3d1b16d1 2023-07-08 jrick }
801 3d1b16d1 2023-07-08 jrick if (ferror(f)) {
802 3d1b16d1 2023-07-08 jrick err = got_error_from_errno("getline");
803 3d1b16d1 2023-07-08 jrick goto done;
804 3d1b16d1 2023-07-08 jrick }
805 3d1b16d1 2023-07-08 jrick
806 3d1b16d1 2023-07-08 jrick dirpath = strdup(path);
807 3d1b16d1 2023-07-08 jrick if (dirpath == NULL) {
808 3d1b16d1 2023-07-08 jrick err = got_error_from_errno("strdup");
809 3d1b16d1 2023-07-08 jrick goto done;
810 3d1b16d1 2023-07-08 jrick }
811 3d1b16d1 2023-07-08 jrick err = got_pathlist_insert(&pe, ignores, dirpath, ignorelist);
812 3d1b16d1 2023-07-08 jrick done:
813 3d1b16d1 2023-07-08 jrick free(line);
814 3d1b16d1 2023-07-08 jrick if (err || pe == NULL) {
815 3d1b16d1 2023-07-08 jrick free(dirpath);
816 3d1b16d1 2023-07-08 jrick got_pathlist_free(ignorelist, GOT_PATHLIST_FREE_PATH);
817 3d1b16d1 2023-07-08 jrick }
818 3d1b16d1 2023-07-08 jrick return err;
819 3d1b16d1 2023-07-08 jrick }
820 3d1b16d1 2023-07-08 jrick
821 3d1b16d1 2023-07-08 jrick static int
822 3d1b16d1 2023-07-08 jrick match_path(const char *pattern, size_t pattern_len, const char *path,
823 3d1b16d1 2023-07-08 jrick int flags)
824 3d1b16d1 2023-07-08 jrick {
825 3d1b16d1 2023-07-08 jrick char buf[PATH_MAX];
826 3d1b16d1 2023-07-08 jrick
827 3d1b16d1 2023-07-08 jrick /*
828 3d1b16d1 2023-07-08 jrick * Trailing slashes signify directories.
829 3d1b16d1 2023-07-08 jrick * Append a * to make such patterns conform to fnmatch rules.
830 3d1b16d1 2023-07-08 jrick */
831 3d1b16d1 2023-07-08 jrick if (pattern_len > 0 && pattern[pattern_len - 1] == '/') {
832 3d1b16d1 2023-07-08 jrick if (snprintf(buf, sizeof(buf), "%s*", pattern) >= sizeof(buf))
833 3d1b16d1 2023-07-08 jrick return FNM_NOMATCH; /* XXX */
834 3d1b16d1 2023-07-08 jrick
835 3d1b16d1 2023-07-08 jrick return fnmatch(buf, path, flags);
836 3d1b16d1 2023-07-08 jrick }
837 3d1b16d1 2023-07-08 jrick
838 3d1b16d1 2023-07-08 jrick return fnmatch(pattern, path, flags);
839 3d1b16d1 2023-07-08 jrick }
840 3d1b16d1 2023-07-08 jrick
841 3d1b16d1 2023-07-08 jrick static int
842 3d1b16d1 2023-07-08 jrick match_ignores(struct got_pathlist_head *ignores, const char *path)
843 3d1b16d1 2023-07-08 jrick {
844 3d1b16d1 2023-07-08 jrick struct got_pathlist_entry *pe;
845 3d1b16d1 2023-07-08 jrick
846 3d1b16d1 2023-07-08 jrick /* Handle patterns which match in all directories. */
847 3d1b16d1 2023-07-08 jrick TAILQ_FOREACH(pe, ignores, entry) {
848 3d1b16d1 2023-07-08 jrick struct got_pathlist_head *ignorelist = pe->data;
849 3d1b16d1 2023-07-08 jrick struct got_pathlist_entry *pi;
850 3d1b16d1 2023-07-08 jrick
851 3d1b16d1 2023-07-08 jrick TAILQ_FOREACH(pi, ignorelist, entry) {
852 3d1b16d1 2023-07-08 jrick const char *p;
853 3d1b16d1 2023-07-08 jrick
854 3d1b16d1 2023-07-08 jrick if (pi->path_len < 3 ||
855 3d1b16d1 2023-07-08 jrick strncmp(pi->path, "**/", 3) != 0)
856 3d1b16d1 2023-07-08 jrick continue;
857 3d1b16d1 2023-07-08 jrick p = path;
858 3d1b16d1 2023-07-08 jrick while (*p) {
859 3d1b16d1 2023-07-08 jrick if (match_path(pi->path + 3,
860 3d1b16d1 2023-07-08 jrick pi->path_len - 3, p,
861 3d1b16d1 2023-07-08 jrick FNM_PATHNAME | FNM_LEADING_DIR)) {
862 3d1b16d1 2023-07-08 jrick /* Retry in next directory. */
863 3d1b16d1 2023-07-08 jrick while (*p && *p != '/')
864 3d1b16d1 2023-07-08 jrick p++;
865 3d1b16d1 2023-07-08 jrick while (*p == '/')
866 3d1b16d1 2023-07-08 jrick p++;
867 3d1b16d1 2023-07-08 jrick continue;
868 3d1b16d1 2023-07-08 jrick }
869 3d1b16d1 2023-07-08 jrick return 1;
870 3d1b16d1 2023-07-08 jrick }
871 3d1b16d1 2023-07-08 jrick }
872 3d1b16d1 2023-07-08 jrick }
873 3d1b16d1 2023-07-08 jrick
874 3d1b16d1 2023-07-08 jrick /*
875 3d1b16d1 2023-07-08 jrick * The ignores pathlist contains ignore lists from children before
876 3d1b16d1 2023-07-08 jrick * parents, so we can find the most specific ignorelist by walking
877 3d1b16d1 2023-07-08 jrick * ignores backwards.
878 3d1b16d1 2023-07-08 jrick */
879 3d1b16d1 2023-07-08 jrick pe = TAILQ_LAST(ignores, got_pathlist_head);
880 3d1b16d1 2023-07-08 jrick while (pe) {
881 3d1b16d1 2023-07-08 jrick if (got_path_is_child(path, pe->path, pe->path_len)) {
882 3d1b16d1 2023-07-08 jrick struct got_pathlist_head *ignorelist = pe->data;
883 3d1b16d1 2023-07-08 jrick struct got_pathlist_entry *pi;
884 3d1b16d1 2023-07-08 jrick TAILQ_FOREACH(pi, ignorelist, entry) {
885 3d1b16d1 2023-07-08 jrick int flags = FNM_LEADING_DIR;
886 3d1b16d1 2023-07-08 jrick if (strstr(pi->path, "/**/") == NULL)
887 3d1b16d1 2023-07-08 jrick flags |= FNM_PATHNAME;
888 3d1b16d1 2023-07-08 jrick if (match_path(pi->path, pi->path_len,
889 3d1b16d1 2023-07-08 jrick path, flags))
890 3d1b16d1 2023-07-08 jrick continue;
891 3d1b16d1 2023-07-08 jrick return 1;
892 3d1b16d1 2023-07-08 jrick }
893 3d1b16d1 2023-07-08 jrick }
894 3d1b16d1 2023-07-08 jrick pe = TAILQ_PREV(pe, got_pathlist_head, entry);
895 3d1b16d1 2023-07-08 jrick }
896 3d1b16d1 2023-07-08 jrick
897 3d1b16d1 2023-07-08 jrick return 0;
898 3d1b16d1 2023-07-08 jrick }
899 3d1b16d1 2023-07-08 jrick
900 3d1b16d1 2023-07-08 jrick static const struct got_error *
901 3d1b16d1 2023-07-08 jrick add_ignores(struct got_pathlist_head *ignores, const char *root_path,
902 3d1b16d1 2023-07-08 jrick const char *path, int dirfd, const char *ignores_filename)
903 3d1b16d1 2023-07-08 jrick {
904 3d1b16d1 2023-07-08 jrick const struct got_error *err = NULL;
905 3d1b16d1 2023-07-08 jrick char *ignorespath;
906 3d1b16d1 2023-07-08 jrick int fd = -1;
907 3d1b16d1 2023-07-08 jrick FILE *ignoresfile = NULL;
908 3d1b16d1 2023-07-08 jrick
909 3d1b16d1 2023-07-08 jrick if (asprintf(&ignorespath, "%s/%s%s%s", root_path, path,
910 3d1b16d1 2023-07-08 jrick path[0] ? "/" : "", ignores_filename) == -1)
911 3d1b16d1 2023-07-08 jrick return got_error_from_errno("asprintf");
912 3d1b16d1 2023-07-08 jrick
913 3d1b16d1 2023-07-08 jrick if (dirfd != -1) {
914 3d1b16d1 2023-07-08 jrick fd = openat(dirfd, ignores_filename,
915 3d1b16d1 2023-07-08 jrick O_RDONLY | O_NOFOLLOW | O_CLOEXEC);
916 3d1b16d1 2023-07-08 jrick if (fd == -1) {
917 3d1b16d1 2023-07-08 jrick if (errno != ENOENT && errno != EACCES)
918 3d1b16d1 2023-07-08 jrick err = got_error_from_errno2("openat",
919 3d1b16d1 2023-07-08 jrick ignorespath);
920 3d1b16d1 2023-07-08 jrick } else {
921 3d1b16d1 2023-07-08 jrick ignoresfile = fdopen(fd, "r");
922 3d1b16d1 2023-07-08 jrick if (ignoresfile == NULL)
923 3d1b16d1 2023-07-08 jrick err = got_error_from_errno2("fdopen",
924 3d1b16d1 2023-07-08 jrick ignorespath);
925 3d1b16d1 2023-07-08 jrick else {
926 3d1b16d1 2023-07-08 jrick fd = -1;
927 3d1b16d1 2023-07-08 jrick err = read_ignores(ignores, path, ignoresfile);
928 3d1b16d1 2023-07-08 jrick }
929 3d1b16d1 2023-07-08 jrick }
930 3d1b16d1 2023-07-08 jrick } else {
931 3d1b16d1 2023-07-08 jrick ignoresfile = fopen(ignorespath, "re");
932 3d1b16d1 2023-07-08 jrick if (ignoresfile == NULL) {
933 3d1b16d1 2023-07-08 jrick if (errno != ENOENT && errno != EACCES)
934 3d1b16d1 2023-07-08 jrick err = got_error_from_errno2("fopen",
935 3d1b16d1 2023-07-08 jrick ignorespath);
936 3d1b16d1 2023-07-08 jrick } else
937 3d1b16d1 2023-07-08 jrick err = read_ignores(ignores, path, ignoresfile);
938 3d1b16d1 2023-07-08 jrick }
939 3d1b16d1 2023-07-08 jrick
940 3d1b16d1 2023-07-08 jrick if (ignoresfile && fclose(ignoresfile) == EOF && err == NULL)
941 3d1b16d1 2023-07-08 jrick err = got_error_from_errno2("fclose", path);
942 3d1b16d1 2023-07-08 jrick if (fd != -1 && close(fd) == -1 && err == NULL)
943 3d1b16d1 2023-07-08 jrick err = got_error_from_errno2("close", path);
944 3d1b16d1 2023-07-08 jrick free(ignorespath);
945 3d1b16d1 2023-07-08 jrick return err;
946 3d1b16d1 2023-07-08 jrick }
947 3d1b16d1 2023-07-08 jrick
948 3d1b16d1 2023-07-08 jrick static const struct got_error *
949 3d1b16d1 2023-07-08 jrick status_new(int *ignore, void *arg, struct dirent *de, const char *parent_path,
950 3d1b16d1 2023-07-08 jrick int dirfd)
951 3d1b16d1 2023-07-08 jrick {
952 3d1b16d1 2023-07-08 jrick const struct got_error *err = NULL;
953 3d1b16d1 2023-07-08 jrick struct diff_dir_cb_arg *a = arg;
954 3d1b16d1 2023-07-08 jrick char *path = NULL;
955 3d1b16d1 2023-07-08 jrick
956 3d1b16d1 2023-07-08 jrick if (ignore != NULL)
957 3d1b16d1 2023-07-08 jrick *ignore = 0;
958 3d1b16d1 2023-07-08 jrick
959 f6b8c3c2 2023-07-24 stsp if (a->cancel_cb) {
960 f6b8c3c2 2023-07-24 stsp err = a->cancel_cb(a->cancel_arg);
961 f6b8c3c2 2023-07-24 stsp if (err)
962 f6b8c3c2 2023-07-24 stsp return err;
963 f6b8c3c2 2023-07-24 stsp }
964 3d1b16d1 2023-07-08 jrick
965 3d1b16d1 2023-07-08 jrick if (parent_path[0]) {
966 3d1b16d1 2023-07-08 jrick if (asprintf(&path, "%s/%s", parent_path, de->d_name) == -1)
967 3d1b16d1 2023-07-08 jrick return got_error_from_errno("asprintf");
968 3d1b16d1 2023-07-08 jrick } else {
969 3d1b16d1 2023-07-08 jrick path = de->d_name;
970 3d1b16d1 2023-07-08 jrick }
971 3d1b16d1 2023-07-08 jrick
972 3d1b16d1 2023-07-08 jrick if (de->d_type == DT_DIR) {
973 3d1b16d1 2023-07-08 jrick if (!a->no_ignores && ignore != NULL &&
974 3d1b16d1 2023-07-08 jrick match_ignores(a->ignores, path))
975 3d1b16d1 2023-07-08 jrick *ignore = 1;
976 3d1b16d1 2023-07-08 jrick } else if (!match_ignores(a->ignores, path) &&
977 3d1b16d1 2023-07-08 jrick got_path_is_child(path, a->status_path, a->status_path_len))
978 3d1b16d1 2023-07-08 jrick err = (*a->status_cb)(a->status_arg, GOT_STATUS_UNVERSIONED,
979 3d1b16d1 2023-07-08 jrick GOT_STATUS_NO_CHANGE, path, NULL, NULL, NULL, -1, NULL);
980 3d1b16d1 2023-07-08 jrick if (parent_path[0])
981 3d1b16d1 2023-07-08 jrick free(path);
982 3d1b16d1 2023-07-08 jrick return err;
983 3d1b16d1 2023-07-08 jrick }
984 3d1b16d1 2023-07-08 jrick
985 3d1b16d1 2023-07-08 jrick static const struct got_error *
986 3d1b16d1 2023-07-08 jrick status_traverse(void *arg, const char *path, int dirfd)
987 3d1b16d1 2023-07-08 jrick {
988 3d1b16d1 2023-07-08 jrick const struct got_error *err = NULL;
989 3d1b16d1 2023-07-08 jrick struct diff_dir_cb_arg *a = arg;
990 3d1b16d1 2023-07-08 jrick
991 3d1b16d1 2023-07-08 jrick if (a->no_ignores)
992 3d1b16d1 2023-07-08 jrick return NULL;
993 3d1b16d1 2023-07-08 jrick
994 3d1b16d1 2023-07-08 jrick err = add_ignores(a->ignores, a->worktree->root_path,
995 3d1b16d1 2023-07-08 jrick path, dirfd, ".cvsignore");
996 3d1b16d1 2023-07-08 jrick if (err)
997 3d1b16d1 2023-07-08 jrick return err;
998 3d1b16d1 2023-07-08 jrick
999 3d1b16d1 2023-07-08 jrick err = add_ignores(a->ignores, a->worktree->root_path, path,
1000 3d1b16d1 2023-07-08 jrick dirfd, ".gitignore");
1001 3d1b16d1 2023-07-08 jrick
1002 3d1b16d1 2023-07-08 jrick return err;
1003 3d1b16d1 2023-07-08 jrick }
1004 3d1b16d1 2023-07-08 jrick
1005 3d1b16d1 2023-07-08 jrick static const struct got_error *
1006 3d1b16d1 2023-07-08 jrick report_single_file_status(const char *path, const char *ondisk_path,
1007 3d1b16d1 2023-07-08 jrick struct got_fileindex *fileindex, got_worktree_status_cb status_cb,
1008 3d1b16d1 2023-07-08 jrick void *status_arg, struct got_repository *repo, int report_unchanged,
1009 3d1b16d1 2023-07-08 jrick struct got_pathlist_head *ignores, int no_ignores)
1010 3d1b16d1 2023-07-08 jrick {
1011 3d1b16d1 2023-07-08 jrick struct got_fileindex_entry *ie;
1012 3d1b16d1 2023-07-08 jrick struct stat sb;
1013 3d1b16d1 2023-07-08 jrick
1014 3d1b16d1 2023-07-08 jrick ie = got_fileindex_entry_get(fileindex, path, strlen(path));
1015 3d1b16d1 2023-07-08 jrick if (ie)
1016 3d1b16d1 2023-07-08 jrick return report_file_status(ie, ondisk_path, -1, NULL,
1017 3d1b16d1 2023-07-08 jrick status_cb, status_arg, repo, report_unchanged);
1018 3d1b16d1 2023-07-08 jrick
1019 3d1b16d1 2023-07-08 jrick if (lstat(ondisk_path, &sb) == -1) {
1020 3d1b16d1 2023-07-08 jrick if (errno != ENOENT)
1021 3d1b16d1 2023-07-08 jrick return got_error_from_errno2("lstat", ondisk_path);
1022 3d1b16d1 2023-07-08 jrick return (*status_cb)(status_arg, GOT_STATUS_NONEXISTENT,
1023 3d1b16d1 2023-07-08 jrick GOT_STATUS_NO_CHANGE, path, NULL, NULL, NULL, -1, NULL);
1024 3d1b16d1 2023-07-08 jrick }
1025 3d1b16d1 2023-07-08 jrick
1026 3d1b16d1 2023-07-08 jrick if (!no_ignores && match_ignores(ignores, path))
1027 3d1b16d1 2023-07-08 jrick return NULL;
1028 3d1b16d1 2023-07-08 jrick
1029 3d1b16d1 2023-07-08 jrick if (S_ISREG(sb.st_mode) || S_ISLNK(sb.st_mode))
1030 3d1b16d1 2023-07-08 jrick return (*status_cb)(status_arg, GOT_STATUS_UNVERSIONED,
1031 3d1b16d1 2023-07-08 jrick GOT_STATUS_NO_CHANGE, path, NULL, NULL, NULL, -1, NULL);
1032 3d1b16d1 2023-07-08 jrick
1033 3d1b16d1 2023-07-08 jrick return NULL;
1034 3d1b16d1 2023-07-08 jrick }
1035 3d1b16d1 2023-07-08 jrick
1036 3d1b16d1 2023-07-08 jrick static const struct got_error *
1037 3d1b16d1 2023-07-08 jrick add_ignores_from_parent_paths(struct got_pathlist_head *ignores,
1038 3d1b16d1 2023-07-08 jrick const char *root_path, const char *path)
1039 3d1b16d1 2023-07-08 jrick {
1040 3d1b16d1 2023-07-08 jrick const struct got_error *err;
1041 3d1b16d1 2023-07-08 jrick char *parent_path, *next_parent_path = NULL;
1042 3d1b16d1 2023-07-08 jrick
1043 3d1b16d1 2023-07-08 jrick err = add_ignores(ignores, root_path, "", -1,
1044 3d1b16d1 2023-07-08 jrick ".cvsignore");
1045 3d1b16d1 2023-07-08 jrick if (err)
1046 3d1b16d1 2023-07-08 jrick return err;
1047 3d1b16d1 2023-07-08 jrick
1048 3d1b16d1 2023-07-08 jrick err = add_ignores(ignores, root_path, "", -1,
1049 3d1b16d1 2023-07-08 jrick ".gitignore");
1050 3d1b16d1 2023-07-08 jrick if (err)
1051 3d1b16d1 2023-07-08 jrick return err;
1052 3d1b16d1 2023-07-08 jrick
1053 3d1b16d1 2023-07-08 jrick err = got_path_dirname(&parent_path, path);
1054 3d1b16d1 2023-07-08 jrick if (err) {
1055 3d1b16d1 2023-07-08 jrick if (err->code == GOT_ERR_BAD_PATH)
1056 3d1b16d1 2023-07-08 jrick return NULL; /* cannot traverse parent */
1057 3d1b16d1 2023-07-08 jrick return err;
1058 3d1b16d1 2023-07-08 jrick }
1059 3d1b16d1 2023-07-08 jrick for (;;) {
1060 3d1b16d1 2023-07-08 jrick err = add_ignores(ignores, root_path, parent_path, -1,
1061 3d1b16d1 2023-07-08 jrick ".cvsignore");
1062 3d1b16d1 2023-07-08 jrick if (err)
1063 3d1b16d1 2023-07-08 jrick break;
1064 3d1b16d1 2023-07-08 jrick err = add_ignores(ignores, root_path, parent_path, -1,
1065 3d1b16d1 2023-07-08 jrick ".gitignore");
1066 3d1b16d1 2023-07-08 jrick if (err)
1067 3d1b16d1 2023-07-08 jrick break;
1068 3d1b16d1 2023-07-08 jrick err = got_path_dirname(&next_parent_path, parent_path);
1069 3d1b16d1 2023-07-08 jrick if (err) {
1070 3d1b16d1 2023-07-08 jrick if (err->code == GOT_ERR_BAD_PATH)
1071 3d1b16d1 2023-07-08 jrick err = NULL; /* traversed everything */
1072 3d1b16d1 2023-07-08 jrick break;
1073 3d1b16d1 2023-07-08 jrick }
1074 3d1b16d1 2023-07-08 jrick if (got_path_is_root_dir(parent_path))
1075 3d1b16d1 2023-07-08 jrick break;
1076 3d1b16d1 2023-07-08 jrick free(parent_path);
1077 3d1b16d1 2023-07-08 jrick parent_path = next_parent_path;
1078 3d1b16d1 2023-07-08 jrick next_parent_path = NULL;
1079 3d1b16d1 2023-07-08 jrick }
1080 3d1b16d1 2023-07-08 jrick
1081 3d1b16d1 2023-07-08 jrick free(parent_path);
1082 3d1b16d1 2023-07-08 jrick free(next_parent_path);
1083 3d1b16d1 2023-07-08 jrick return err;
1084 3d1b16d1 2023-07-08 jrick }
1085 3d1b16d1 2023-07-08 jrick
1086 3d1b16d1 2023-07-08 jrick struct find_missing_children_args {
1087 3d1b16d1 2023-07-08 jrick const char *parent_path;
1088 3d1b16d1 2023-07-08 jrick size_t parent_len;
1089 3d1b16d1 2023-07-08 jrick struct got_pathlist_head *children;
1090 3d1b16d1 2023-07-08 jrick got_cancel_cb cancel_cb;
1091 3d1b16d1 2023-07-08 jrick void *cancel_arg;
1092 3d1b16d1 2023-07-08 jrick };
1093 3d1b16d1 2023-07-08 jrick
1094 3d1b16d1 2023-07-08 jrick static const struct got_error *
1095 3d1b16d1 2023-07-08 jrick find_missing_children(void *arg, struct got_fileindex_entry *ie)
1096 3d1b16d1 2023-07-08 jrick {
1097 3d1b16d1 2023-07-08 jrick const struct got_error *err = NULL;
1098 3d1b16d1 2023-07-08 jrick struct find_missing_children_args *a = arg;
1099 3d1b16d1 2023-07-08 jrick
1100 3d1b16d1 2023-07-08 jrick if (a->cancel_cb) {
1101 3d1b16d1 2023-07-08 jrick err = a->cancel_cb(a->cancel_arg);
1102 3d1b16d1 2023-07-08 jrick if (err)
1103 3d1b16d1 2023-07-08 jrick return err;
1104 3d1b16d1 2023-07-08 jrick }
1105 3d1b16d1 2023-07-08 jrick
1106 3d1b16d1 2023-07-08 jrick if (got_path_is_child(ie->path, a->parent_path, a->parent_len))
1107 3d1b16d1 2023-07-08 jrick err = got_pathlist_append(a->children, ie->path, NULL);
1108 3d1b16d1 2023-07-08 jrick
1109 3d1b16d1 2023-07-08 jrick return err;
1110 3d1b16d1 2023-07-08 jrick }
1111 3d1b16d1 2023-07-08 jrick
1112 3d1b16d1 2023-07-08 jrick static const struct got_error *
1113 3d1b16d1 2023-07-08 jrick report_children(struct got_pathlist_head *children,
1114 3d1b16d1 2023-07-08 jrick struct got_worktree *worktree, struct got_fileindex *fileindex,
1115 3d1b16d1 2023-07-08 jrick struct got_repository *repo, int is_root_dir, int report_unchanged,
1116 3d1b16d1 2023-07-08 jrick struct got_pathlist_head *ignores, int no_ignores,
1117 3d1b16d1 2023-07-08 jrick got_worktree_status_cb status_cb, void *status_arg,
1118 3d1b16d1 2023-07-08 jrick got_cancel_cb cancel_cb, void *cancel_arg)
1119 3d1b16d1 2023-07-08 jrick {
1120 3d1b16d1 2023-07-08 jrick const struct got_error *err = NULL;
1121 3d1b16d1 2023-07-08 jrick struct got_pathlist_entry *pe;
1122 3d1b16d1 2023-07-08 jrick char *ondisk_path = NULL;
1123 3d1b16d1 2023-07-08 jrick
1124 3d1b16d1 2023-07-08 jrick TAILQ_FOREACH(pe, children, entry) {
1125 3d1b16d1 2023-07-08 jrick if (cancel_cb) {
1126 3d1b16d1 2023-07-08 jrick err = cancel_cb(cancel_arg);
1127 3d1b16d1 2023-07-08 jrick if (err)
1128 3d1b16d1 2023-07-08 jrick break;
1129 3d1b16d1 2023-07-08 jrick }
1130 3d1b16d1 2023-07-08 jrick
1131 3d1b16d1 2023-07-08 jrick if (asprintf(&ondisk_path, "%s%s%s", worktree->root_path,
1132 3d1b16d1 2023-07-08 jrick !is_root_dir ? "/" : "", pe->path) == -1) {
1133 3d1b16d1 2023-07-08 jrick err = got_error_from_errno("asprintf");
1134 3d1b16d1 2023-07-08 jrick ondisk_path = NULL;
1135 3d1b16d1 2023-07-08 jrick break;
1136 3d1b16d1 2023-07-08 jrick }
1137 3d1b16d1 2023-07-08 jrick
1138 3d1b16d1 2023-07-08 jrick err = report_single_file_status(pe->path, ondisk_path,
1139 3d1b16d1 2023-07-08 jrick fileindex, status_cb, status_arg, repo, report_unchanged,
1140 3d1b16d1 2023-07-08 jrick ignores, no_ignores);
1141 3d1b16d1 2023-07-08 jrick if (err)
1142 3d1b16d1 2023-07-08 jrick break;
1143 3d1b16d1 2023-07-08 jrick
1144 3d1b16d1 2023-07-08 jrick free(ondisk_path);
1145 3d1b16d1 2023-07-08 jrick ondisk_path = NULL;
1146 3d1b16d1 2023-07-08 jrick }
1147 3d1b16d1 2023-07-08 jrick
1148 3d1b16d1 2023-07-08 jrick free(ondisk_path);
1149 3d1b16d1 2023-07-08 jrick return err;
1150 3d1b16d1 2023-07-08 jrick }
1151 3d1b16d1 2023-07-08 jrick
1152 3d1b16d1 2023-07-08 jrick static const struct got_error *
1153 3d1b16d1 2023-07-08 jrick worktree_status(struct got_worktree *worktree, const char *path,
1154 3d1b16d1 2023-07-08 jrick struct got_fileindex *fileindex, struct got_repository *repo,
1155 3d1b16d1 2023-07-08 jrick got_worktree_status_cb status_cb, void *status_arg,
1156 3d1b16d1 2023-07-08 jrick got_cancel_cb cancel_cb, void *cancel_arg, int no_ignores,
1157 3d1b16d1 2023-07-08 jrick int report_unchanged)
1158 3d1b16d1 2023-07-08 jrick {
1159 3d1b16d1 2023-07-08 jrick const struct got_error *err = NULL;
1160 3d1b16d1 2023-07-08 jrick int fd = -1;
1161 3d1b16d1 2023-07-08 jrick struct got_fileindex_diff_dir_cb fdiff_cb;
1162 3d1b16d1 2023-07-08 jrick struct diff_dir_cb_arg arg;
1163 3d1b16d1 2023-07-08 jrick char *ondisk_path = NULL;
1164 3d1b16d1 2023-07-08 jrick struct got_pathlist_head ignores, missing_children;
1165 3d1b16d1 2023-07-08 jrick struct got_fileindex_entry *ie;
1166 3d1b16d1 2023-07-08 jrick
1167 3d1b16d1 2023-07-08 jrick TAILQ_INIT(&ignores);
1168 3d1b16d1 2023-07-08 jrick TAILQ_INIT(&missing_children);
1169 3d1b16d1 2023-07-08 jrick
1170 3d1b16d1 2023-07-08 jrick if (asprintf(&ondisk_path, "%s%s%s",
1171 3d1b16d1 2023-07-08 jrick worktree->root_path, path[0] ? "/" : "", path) == -1)
1172 3d1b16d1 2023-07-08 jrick return got_error_from_errno("asprintf");
1173 3d1b16d1 2023-07-08 jrick
1174 3d1b16d1 2023-07-08 jrick ie = got_fileindex_entry_get(fileindex, path, strlen(path));
1175 3d1b16d1 2023-07-08 jrick if (ie) {
1176 3d1b16d1 2023-07-08 jrick err = report_single_file_status(path, ondisk_path,
1177 3d1b16d1 2023-07-08 jrick fileindex, status_cb, status_arg, repo,
1178 3d1b16d1 2023-07-08 jrick report_unchanged, &ignores, no_ignores);
1179 3d1b16d1 2023-07-08 jrick goto done;
1180 3d1b16d1 2023-07-08 jrick } else {
1181 3d1b16d1 2023-07-08 jrick struct find_missing_children_args fmca;
1182 3d1b16d1 2023-07-08 jrick fmca.parent_path = path;
1183 3d1b16d1 2023-07-08 jrick fmca.parent_len = strlen(path);
1184 3d1b16d1 2023-07-08 jrick fmca.children = &missing_children;
1185 3d1b16d1 2023-07-08 jrick fmca.cancel_cb = cancel_cb;
1186 3d1b16d1 2023-07-08 jrick fmca.cancel_arg = cancel_arg;
1187 3d1b16d1 2023-07-08 jrick err = got_fileindex_for_each_entry_safe(fileindex,
1188 3d1b16d1 2023-07-08 jrick find_missing_children, &fmca);
1189 3d1b16d1 2023-07-08 jrick if (err)
1190 3d1b16d1 2023-07-08 jrick goto done;
1191 3d1b16d1 2023-07-08 jrick }
1192 3d1b16d1 2023-07-08 jrick
1193 3d1b16d1 2023-07-08 jrick fd = open(ondisk_path, O_RDONLY | O_NOFOLLOW | O_DIRECTORY | O_CLOEXEC);
1194 3d1b16d1 2023-07-08 jrick if (fd == -1) {
1195 3d1b16d1 2023-07-08 jrick if (errno != ENOTDIR && errno != ENOENT && errno != EACCES &&
1196 3d1b16d1 2023-07-08 jrick !got_err_open_nofollow_on_symlink())
1197 3d1b16d1 2023-07-08 jrick err = got_error_from_errno2("open", ondisk_path);
1198 3d1b16d1 2023-07-08 jrick else {
1199 3d1b16d1 2023-07-08 jrick if (!no_ignores) {
1200 3d1b16d1 2023-07-08 jrick err = add_ignores_from_parent_paths(&ignores,
1201 3d1b16d1 2023-07-08 jrick worktree->root_path, ondisk_path);
1202 3d1b16d1 2023-07-08 jrick if (err)
1203 3d1b16d1 2023-07-08 jrick goto done;
1204 3d1b16d1 2023-07-08 jrick }
1205 3d1b16d1 2023-07-08 jrick if (TAILQ_EMPTY(&missing_children)) {
1206 3d1b16d1 2023-07-08 jrick err = report_single_file_status(path,
1207 3d1b16d1 2023-07-08 jrick ondisk_path, fileindex,
1208 3d1b16d1 2023-07-08 jrick status_cb, status_arg, repo,
1209 3d1b16d1 2023-07-08 jrick report_unchanged, &ignores, no_ignores);
1210 3d1b16d1 2023-07-08 jrick if (err)
1211 3d1b16d1 2023-07-08 jrick goto done;
1212 3d1b16d1 2023-07-08 jrick } else {
1213 3d1b16d1 2023-07-08 jrick err = report_children(&missing_children,
1214 3d1b16d1 2023-07-08 jrick worktree, fileindex, repo,
1215 3d1b16d1 2023-07-08 jrick (path[0] == '\0'), report_unchanged,
1216 3d1b16d1 2023-07-08 jrick &ignores, no_ignores,
1217 3d1b16d1 2023-07-08 jrick status_cb, status_arg,
1218 3d1b16d1 2023-07-08 jrick cancel_cb, cancel_arg);
1219 3d1b16d1 2023-07-08 jrick if (err)
1220 3d1b16d1 2023-07-08 jrick goto done;
1221 3d1b16d1 2023-07-08 jrick }
1222 3d1b16d1 2023-07-08 jrick }
1223 3d1b16d1 2023-07-08 jrick } else {
1224 3d1b16d1 2023-07-08 jrick fdiff_cb.diff_old_new = status_old_new;
1225 3d1b16d1 2023-07-08 jrick fdiff_cb.diff_old = status_old;
1226 3d1b16d1 2023-07-08 jrick fdiff_cb.diff_new = status_new;
1227 3d1b16d1 2023-07-08 jrick fdiff_cb.diff_traverse = status_traverse;
1228 3d1b16d1 2023-07-08 jrick arg.fileindex = fileindex;
1229 3d1b16d1 2023-07-08 jrick arg.worktree = worktree;
1230 3d1b16d1 2023-07-08 jrick arg.status_path = path;
1231 3d1b16d1 2023-07-08 jrick arg.status_path_len = strlen(path);
1232 3d1b16d1 2023-07-08 jrick arg.repo = repo;
1233 3d1b16d1 2023-07-08 jrick arg.status_cb = status_cb;
1234 3d1b16d1 2023-07-08 jrick arg.status_arg = status_arg;
1235 3d1b16d1 2023-07-08 jrick arg.cancel_cb = cancel_cb;
1236 3d1b16d1 2023-07-08 jrick arg.cancel_arg = cancel_arg;
1237 3d1b16d1 2023-07-08 jrick arg.report_unchanged = report_unchanged;
1238 3d1b16d1 2023-07-08 jrick arg.no_ignores = no_ignores;
1239 3d1b16d1 2023-07-08 jrick if (!no_ignores) {
1240 3d1b16d1 2023-07-08 jrick err = add_ignores_from_parent_paths(&ignores,
1241 3d1b16d1 2023-07-08 jrick worktree->root_path, path);
1242 3d1b16d1 2023-07-08 jrick if (err)
1243 3d1b16d1 2023-07-08 jrick goto done;
1244 3d1b16d1 2023-07-08 jrick }
1245 3d1b16d1 2023-07-08 jrick arg.ignores = &ignores;
1246 3d1b16d1 2023-07-08 jrick err = got_fileindex_diff_dir(fileindex, fd,
1247 3d1b16d1 2023-07-08 jrick worktree->root_path, path, repo, &fdiff_cb, &arg);
1248 3d1b16d1 2023-07-08 jrick }
1249 3d1b16d1 2023-07-08 jrick done:
1250 3d1b16d1 2023-07-08 jrick free_ignores(&ignores);
1251 3d1b16d1 2023-07-08 jrick if (fd != -1 && close(fd) == -1 && err == NULL)
1252 3d1b16d1 2023-07-08 jrick err = got_error_from_errno("close");
1253 3d1b16d1 2023-07-08 jrick free(ondisk_path);
1254 3d1b16d1 2023-07-08 jrick return err;
1255 3d1b16d1 2023-07-08 jrick }
1256 3d1b16d1 2023-07-08 jrick
1257 3d1b16d1 2023-07-08 jrick static void
1258 3d1b16d1 2023-07-08 jrick free_commitable(struct got_commitable *ct)
1259 3d1b16d1 2023-07-08 jrick {
1260 3d1b16d1 2023-07-08 jrick free(ct->path);
1261 3d1b16d1 2023-07-08 jrick free(ct->in_repo_path);
1262 3d1b16d1 2023-07-08 jrick free(ct->ondisk_path);
1263 3d1b16d1 2023-07-08 jrick free(ct->blob_id);
1264 3d1b16d1 2023-07-08 jrick free(ct->base_blob_id);
1265 3d1b16d1 2023-07-08 jrick free(ct->staged_blob_id);
1266 3d1b16d1 2023-07-08 jrick free(ct->base_commit_id);
1267 3d1b16d1 2023-07-08 jrick free(ct);
1268 3d1b16d1 2023-07-08 jrick }
1269 3d1b16d1 2023-07-08 jrick
1270 3d1b16d1 2023-07-08 jrick struct collect_commitables_arg {
1271 3d1b16d1 2023-07-08 jrick struct got_pathlist_head *commitable_paths;
1272 3d1b16d1 2023-07-08 jrick struct got_repository *repo;
1273 3d1b16d1 2023-07-08 jrick struct got_worktree *worktree;
1274 3d1b16d1 2023-07-08 jrick struct got_fileindex *fileindex;
1275 3d1b16d1 2023-07-08 jrick int have_staged_files;
1276 3d1b16d1 2023-07-08 jrick int allow_bad_symlinks;
1277 3d1b16d1 2023-07-08 jrick int diff_header_shown;
1278 3d1b16d1 2023-07-08 jrick int commit_conflicts;
1279 3d1b16d1 2023-07-08 jrick FILE *diff_outfile;
1280 3d1b16d1 2023-07-08 jrick FILE *f1;
1281 3d1b16d1 2023-07-08 jrick FILE *f2;
1282 3d1b16d1 2023-07-08 jrick };
1283 3d1b16d1 2023-07-08 jrick
1284 3d1b16d1 2023-07-08 jrick /*
1285 3d1b16d1 2023-07-08 jrick * Create a file which contains the target path of a symlink so we can feed
1286 3d1b16d1 2023-07-08 jrick * it as content to the diff engine.
1287 3d1b16d1 2023-07-08 jrick */
1288 3d1b16d1 2023-07-08 jrick static const struct got_error *
1289 3d1b16d1 2023-07-08 jrick get_symlink_target_file(int *fd, int dirfd, const char *de_name,
1290 3d1b16d1 2023-07-08 jrick const char *abspath)
1291 3d1b16d1 2023-07-08 jrick {
1292 3d1b16d1 2023-07-08 jrick const struct got_error *err = NULL;
1293 3d1b16d1 2023-07-08 jrick char target_path[PATH_MAX];
1294 3d1b16d1 2023-07-08 jrick ssize_t target_len, outlen;
1295 3d1b16d1 2023-07-08 jrick
1296 3d1b16d1 2023-07-08 jrick *fd = -1;
1297 3d1b16d1 2023-07-08 jrick
1298 3d1b16d1 2023-07-08 jrick if (dirfd != -1) {
1299 3d1b16d1 2023-07-08 jrick target_len = readlinkat(dirfd, de_name, target_path, PATH_MAX);
1300 3d1b16d1 2023-07-08 jrick if (target_len == -1)
1301 3d1b16d1 2023-07-08 jrick return got_error_from_errno2("readlinkat", abspath);
1302 3d1b16d1 2023-07-08 jrick } else {
1303 3d1b16d1 2023-07-08 jrick target_len = readlink(abspath, target_path, PATH_MAX);
1304 3d1b16d1 2023-07-08 jrick if (target_len == -1)
1305 3d1b16d1 2023-07-08 jrick return got_error_from_errno2("readlink", abspath);
1306 3d1b16d1 2023-07-08 jrick }
1307 3d1b16d1 2023-07-08 jrick
1308 3d1b16d1 2023-07-08 jrick *fd = got_opentempfd();
1309 3d1b16d1 2023-07-08 jrick if (*fd == -1)
1310 3d1b16d1 2023-07-08 jrick return got_error_from_errno("got_opentempfd");
1311 3d1b16d1 2023-07-08 jrick
1312 3d1b16d1 2023-07-08 jrick outlen = write(*fd, target_path, target_len);
1313 3d1b16d1 2023-07-08 jrick if (outlen == -1) {
1314 3d1b16d1 2023-07-08 jrick err = got_error_from_errno("got_opentempfd");
1315 3d1b16d1 2023-07-08 jrick goto done;
1316 3d1b16d1 2023-07-08 jrick }
1317 3d1b16d1 2023-07-08 jrick
1318 3d1b16d1 2023-07-08 jrick if (lseek(*fd, 0, SEEK_SET) == -1) {
1319 3d1b16d1 2023-07-08 jrick err = got_error_from_errno2("lseek", abspath);
1320 3d1b16d1 2023-07-08 jrick goto done;
1321 3d1b16d1 2023-07-08 jrick }
1322 3d1b16d1 2023-07-08 jrick done:
1323 3d1b16d1 2023-07-08 jrick if (err) {
1324 3d1b16d1 2023-07-08 jrick close(*fd);
1325 3d1b16d1 2023-07-08 jrick *fd = -1;
1326 3d1b16d1 2023-07-08 jrick }
1327 3d1b16d1 2023-07-08 jrick return err;
1328 3d1b16d1 2023-07-08 jrick }
1329 3d1b16d1 2023-07-08 jrick
1330 3d1b16d1 2023-07-08 jrick static const struct got_error *
1331 3d1b16d1 2023-07-08 jrick append_ct_diff(struct got_commitable *ct, int *diff_header_shown,
1332 3d1b16d1 2023-07-08 jrick FILE *diff_outfile, FILE *f1, FILE *f2, int dirfd, const char *de_name,
1333 3d1b16d1 2023-07-08 jrick int diff_staged, struct got_repository *repo, struct got_worktree *worktree)
1334 3d1b16d1 2023-07-08 jrick {
1335 3d1b16d1 2023-07-08 jrick const struct got_error *err = NULL;
1336 3d1b16d1 2023-07-08 jrick struct got_blob_object *blob1 = NULL;
1337 3d1b16d1 2023-07-08 jrick int fd = -1, fd1 = -1, fd2 = -1;
1338 3d1b16d1 2023-07-08 jrick FILE *ondisk_file = NULL;
1339 3d1b16d1 2023-07-08 jrick char *label1 = NULL;
1340 3d1b16d1 2023-07-08 jrick struct stat sb;
1341 3d1b16d1 2023-07-08 jrick off_t size1 = 0;
1342 3d1b16d1 2023-07-08 jrick int f2_exists = 0;
1343 3d1b16d1 2023-07-08 jrick char *id_str = NULL;
1344 3d1b16d1 2023-07-08 jrick
1345 3d1b16d1 2023-07-08 jrick memset(&sb, 0, sizeof(sb));
1346 3d1b16d1 2023-07-08 jrick
1347 3d1b16d1 2023-07-08 jrick if (diff_staged) {
1348 3d1b16d1 2023-07-08 jrick if (ct->staged_status != GOT_STATUS_MODIFY &&
1349 3d1b16d1 2023-07-08 jrick ct->staged_status != GOT_STATUS_ADD &&
1350 3d1b16d1 2023-07-08 jrick ct->staged_status != GOT_STATUS_DELETE)
1351 3d1b16d1 2023-07-08 jrick return NULL;
1352 3d1b16d1 2023-07-08 jrick } else {
1353 3d1b16d1 2023-07-08 jrick if (ct->status != GOT_STATUS_MODIFY &&
1354 3d1b16d1 2023-07-08 jrick ct->status != GOT_STATUS_ADD &&
1355 3d1b16d1 2023-07-08 jrick ct->status != GOT_STATUS_DELETE &&
1356 3d1b16d1 2023-07-08 jrick ct->status != GOT_STATUS_CONFLICT)
1357 3d1b16d1 2023-07-08 jrick return NULL;
1358 3d1b16d1 2023-07-08 jrick }
1359 3d1b16d1 2023-07-08 jrick
1360 3d1b16d1 2023-07-08 jrick err = got_opentemp_truncate(f1);
1361 3d1b16d1 2023-07-08 jrick if (err)
1362 3d1b16d1 2023-07-08 jrick return got_error_from_errno("got_opentemp_truncate");
1363 3d1b16d1 2023-07-08 jrick err = got_opentemp_truncate(f2);
1364 3d1b16d1 2023-07-08 jrick if (err)
1365 3d1b16d1 2023-07-08 jrick return got_error_from_errno("got_opentemp_truncate");
1366 3d1b16d1 2023-07-08 jrick
1367 3d1b16d1 2023-07-08 jrick if (!*diff_header_shown) {
1368 3d1b16d1 2023-07-08 jrick err = got_object_id_str(&id_str, worktree->base_commit_id);
1369 3d1b16d1 2023-07-08 jrick if (err)
1370 3d1b16d1 2023-07-08 jrick return err;
1371 3d1b16d1 2023-07-08 jrick fprintf(diff_outfile, "diff %s%s\n", diff_staged ? "-s " : "",
1372 3d1b16d1 2023-07-08 jrick got_worktree_get_root_path(worktree));
1373 3d1b16d1 2023-07-08 jrick fprintf(diff_outfile, "commit - %s\n", id_str);
1374 3d1b16d1 2023-07-08 jrick fprintf(diff_outfile, "path + %s%s\n",
1375 3d1b16d1 2023-07-08 jrick got_worktree_get_root_path(worktree),
1376 3d1b16d1 2023-07-08 jrick diff_staged ? " (staged changes)" : "");
1377 3d1b16d1 2023-07-08 jrick *diff_header_shown = 1;
1378 3d1b16d1 2023-07-08 jrick }
1379 3d1b16d1 2023-07-08 jrick
1380 3d1b16d1 2023-07-08 jrick if (diff_staged) {
1381 3d1b16d1 2023-07-08 jrick const char *label1 = NULL, *label2 = NULL;
1382 3d1b16d1 2023-07-08 jrick switch (ct->staged_status) {
1383 3d1b16d1 2023-07-08 jrick case GOT_STATUS_MODIFY:
1384 3d1b16d1 2023-07-08 jrick label1 = ct->path;
1385 3d1b16d1 2023-07-08 jrick label2 = ct->path;
1386 3d1b16d1 2023-07-08 jrick break;
1387 3d1b16d1 2023-07-08 jrick case GOT_STATUS_ADD:
1388 3d1b16d1 2023-07-08 jrick label2 = ct->path;
1389 3d1b16d1 2023-07-08 jrick break;
1390 3d1b16d1 2023-07-08 jrick case GOT_STATUS_DELETE:
1391 3d1b16d1 2023-07-08 jrick label1 = ct->path;
1392 3d1b16d1 2023-07-08 jrick break;
1393 3d1b16d1 2023-07-08 jrick default:
1394 3d1b16d1 2023-07-08 jrick return got_error(GOT_ERR_FILE_STATUS);
1395 3d1b16d1 2023-07-08 jrick }
1396 3d1b16d1 2023-07-08 jrick fd1 = got_opentempfd();
1397 3d1b16d1 2023-07-08 jrick if (fd1 == -1) {
1398 3d1b16d1 2023-07-08 jrick err = got_error_from_errno("got_opentempfd");
1399 3d1b16d1 2023-07-08 jrick goto done;
1400 3d1b16d1 2023-07-08 jrick }
1401 3d1b16d1 2023-07-08 jrick fd2 = got_opentempfd();
1402 3d1b16d1 2023-07-08 jrick if (fd2 == -1) {
1403 3d1b16d1 2023-07-08 jrick err = got_error_from_errno("got_opentempfd");
1404 3d1b16d1 2023-07-08 jrick goto done;
1405 3d1b16d1 2023-07-08 jrick }
1406 3d1b16d1 2023-07-08 jrick err = got_diff_objects_as_blobs(NULL, NULL, f1, f2,
1407 3d1b16d1 2023-07-08 jrick fd1, fd2, ct->base_blob_id, ct->staged_blob_id,
1408 3d1b16d1 2023-07-08 jrick label1, label2, GOT_DIFF_ALGORITHM_PATIENCE, 3, 0, 0,
1409 3d1b16d1 2023-07-08 jrick NULL, repo, diff_outfile);
1410 3d1b16d1 2023-07-08 jrick goto done;
1411 3d1b16d1 2023-07-08 jrick }
1412 3d1b16d1 2023-07-08 jrick
1413 3d1b16d1 2023-07-08 jrick fd1 = got_opentempfd();
1414 3d1b16d1 2023-07-08 jrick if (fd1 == -1) {
1415 3d1b16d1 2023-07-08 jrick err = got_error_from_errno("got_opentempfd");
1416 3d1b16d1 2023-07-08 jrick goto done;
1417 3d1b16d1 2023-07-08 jrick }
1418 3d1b16d1 2023-07-08 jrick
1419 3d1b16d1 2023-07-08 jrick if (ct->status != GOT_STATUS_ADD) {
1420 3d1b16d1 2023-07-08 jrick err = got_object_open_as_blob(&blob1, repo, ct->base_blob_id,
1421 3d1b16d1 2023-07-08 jrick 8192, fd1);
1422 3d1b16d1 2023-07-08 jrick if (err)
1423 3d1b16d1 2023-07-08 jrick goto done;
1424 3d1b16d1 2023-07-08 jrick }
1425 3d1b16d1 2023-07-08 jrick
1426 3d1b16d1 2023-07-08 jrick if (ct->status != GOT_STATUS_DELETE) {
1427 3d1b16d1 2023-07-08 jrick if (dirfd != -1) {
1428 3d1b16d1 2023-07-08 jrick fd = openat(dirfd, de_name,
1429 3d1b16d1 2023-07-08 jrick O_RDONLY | O_NOFOLLOW | O_CLOEXEC);
1430 3d1b16d1 2023-07-08 jrick if (fd == -1) {
1431 3d1b16d1 2023-07-08 jrick if (!got_err_open_nofollow_on_symlink()) {
1432 3d1b16d1 2023-07-08 jrick err = got_error_from_errno2("openat",
1433 3d1b16d1 2023-07-08 jrick ct->ondisk_path);
1434 3d1b16d1 2023-07-08 jrick goto done;
1435 3d1b16d1 2023-07-08 jrick }
1436 3d1b16d1 2023-07-08 jrick err = get_symlink_target_file(&fd, dirfd,
1437 3d1b16d1 2023-07-08 jrick de_name, ct->ondisk_path);
1438 3d1b16d1 2023-07-08 jrick if (err)
1439 3d1b16d1 2023-07-08 jrick goto done;
1440 3d1b16d1 2023-07-08 jrick }
1441 3d1b16d1 2023-07-08 jrick } else {
1442 3d1b16d1 2023-07-08 jrick fd = open(ct->ondisk_path,
1443 3d1b16d1 2023-07-08 jrick O_RDONLY | O_NOFOLLOW | O_CLOEXEC);
1444 3d1b16d1 2023-07-08 jrick if (fd == -1) {
1445 3d1b16d1 2023-07-08 jrick if (!got_err_open_nofollow_on_symlink()) {
1446 3d1b16d1 2023-07-08 jrick err = got_error_from_errno2("open",
1447 3d1b16d1 2023-07-08 jrick ct->ondisk_path);
1448 3d1b16d1 2023-07-08 jrick goto done;
1449 3d1b16d1 2023-07-08 jrick }
1450 3d1b16d1 2023-07-08 jrick err = get_symlink_target_file(&fd, dirfd,
1451 3d1b16d1 2023-07-08 jrick de_name, ct->ondisk_path);
1452 3d1b16d1 2023-07-08 jrick if (err)
1453 3d1b16d1 2023-07-08 jrick goto done;
1454 3d1b16d1 2023-07-08 jrick }
1455 3d1b16d1 2023-07-08 jrick }
1456 3d1b16d1 2023-07-08 jrick if (fstatat(fd, ct->ondisk_path, &sb,
1457 3d1b16d1 2023-07-08 jrick AT_SYMLINK_NOFOLLOW) == -1) {
1458 3d1b16d1 2023-07-08 jrick err = got_error_from_errno2("fstatat", ct->ondisk_path);
1459 3d1b16d1 2023-07-08 jrick goto done;
1460 3d1b16d1 2023-07-08 jrick }
1461 3d1b16d1 2023-07-08 jrick ondisk_file = fdopen(fd, "r");
1462 3d1b16d1 2023-07-08 jrick if (ondisk_file == NULL) {
1463 3d1b16d1 2023-07-08 jrick err = got_error_from_errno2("fdopen", ct->ondisk_path);
1464 3d1b16d1 2023-07-08 jrick goto done;
1465 3d1b16d1 2023-07-08 jrick }
1466 3d1b16d1 2023-07-08 jrick fd = -1;
1467 3d1b16d1 2023-07-08 jrick f2_exists = 1;
1468 3d1b16d1 2023-07-08 jrick }
1469 3d1b16d1 2023-07-08 jrick
1470 3d1b16d1 2023-07-08 jrick if (blob1) {
1471 3d1b16d1 2023-07-08 jrick err = got_object_blob_dump_to_file(&size1, NULL, NULL,
1472 3d1b16d1 2023-07-08 jrick f1, blob1);
1473 3d1b16d1 2023-07-08 jrick if (err)
1474 3d1b16d1 2023-07-08 jrick goto done;
1475 3d1b16d1 2023-07-08 jrick }
1476 3d1b16d1 2023-07-08 jrick
1477 3d1b16d1 2023-07-08 jrick err = got_diff_blob_file(blob1, f1, size1, label1,
1478 3d1b16d1 2023-07-08 jrick ondisk_file ? ondisk_file : f2, f2_exists, &sb, ct->path,
1479 3d1b16d1 2023-07-08 jrick GOT_DIFF_ALGORITHM_PATIENCE, 3, 0, 0, NULL, diff_outfile);
1480 3d1b16d1 2023-07-08 jrick done:
1481 3d1b16d1 2023-07-08 jrick if (fd1 != -1 && close(fd1) == -1 && err == NULL)
1482 3d1b16d1 2023-07-08 jrick err = got_error_from_errno("close");
1483 3d1b16d1 2023-07-08 jrick if (fd2 != -1 && close(fd2) == -1 && err == NULL)
1484 3d1b16d1 2023-07-08 jrick err = got_error_from_errno("close");
1485 3d1b16d1 2023-07-08 jrick if (blob1)
1486 3d1b16d1 2023-07-08 jrick got_object_blob_close(blob1);
1487 3d1b16d1 2023-07-08 jrick if (fd != -1 && close(fd) == -1 && err == NULL)
1488 3d1b16d1 2023-07-08 jrick err = got_error_from_errno("close");
1489 3d1b16d1 2023-07-08 jrick if (ondisk_file && fclose(ondisk_file) == EOF && err == NULL)
1490 3d1b16d1 2023-07-08 jrick err = got_error_from_errno("fclose");
1491 3d1b16d1 2023-07-08 jrick return err;
1492 3d1b16d1 2023-07-08 jrick }
1493 3d1b16d1 2023-07-08 jrick
1494 3d1b16d1 2023-07-08 jrick static const struct got_error *
1495 3d1b16d1 2023-07-08 jrick collect_commitables(void *arg, unsigned char status,
1496 3d1b16d1 2023-07-08 jrick unsigned char staged_status, const char *relpath,
1497 3d1b16d1 2023-07-08 jrick struct got_object_id *blob_id, struct got_object_id *staged_blob_id,
1498 3d1b16d1 2023-07-08 jrick struct got_object_id *commit_id, int dirfd, const char *de_name)
1499 3d1b16d1 2023-07-08 jrick {
1500 3d1b16d1 2023-07-08 jrick struct collect_commitables_arg *a = arg;
1501 3d1b16d1 2023-07-08 jrick const struct got_error *err = NULL;
1502 3d1b16d1 2023-07-08 jrick struct got_commitable *ct = NULL;
1503 3d1b16d1 2023-07-08 jrick struct got_pathlist_entry *new = NULL;
1504 3d1b16d1 2023-07-08 jrick char *parent_path = NULL, *path = NULL;
1505 3d1b16d1 2023-07-08 jrick struct stat sb;
1506 3d1b16d1 2023-07-08 jrick
1507 3d1b16d1 2023-07-08 jrick if (a->have_staged_files) {
1508 3d1b16d1 2023-07-08 jrick if (staged_status != GOT_STATUS_MODIFY &&
1509 3d1b16d1 2023-07-08 jrick staged_status != GOT_STATUS_ADD &&
1510 3d1b16d1 2023-07-08 jrick staged_status != GOT_STATUS_DELETE)
1511 3d1b16d1 2023-07-08 jrick return NULL;
1512 3d1b16d1 2023-07-08 jrick } else {
1513 3d1b16d1 2023-07-08 jrick if (status == GOT_STATUS_CONFLICT && !a->commit_conflicts) {
1514 3d1b16d1 2023-07-08 jrick printf("C %s\n", relpath);
1515 3d1b16d1 2023-07-08 jrick return got_error(GOT_ERR_COMMIT_CONFLICT);
1516 3d1b16d1 2023-07-08 jrick }
1517 3d1b16d1 2023-07-08 jrick
1518 3d1b16d1 2023-07-08 jrick if (status != GOT_STATUS_MODIFY &&
1519 3d1b16d1 2023-07-08 jrick status != GOT_STATUS_MODE_CHANGE &&
1520 3d1b16d1 2023-07-08 jrick status != GOT_STATUS_ADD &&
1521 3d1b16d1 2023-07-08 jrick status != GOT_STATUS_DELETE &&
1522 3d1b16d1 2023-07-08 jrick status != GOT_STATUS_CONFLICT)
1523 3d1b16d1 2023-07-08 jrick return NULL;
1524 3d1b16d1 2023-07-08 jrick }
1525 3d1b16d1 2023-07-08 jrick
1526 3d1b16d1 2023-07-08 jrick if (asprintf(&path, "/%s", relpath) == -1) {
1527 3d1b16d1 2023-07-08 jrick err = got_error_from_errno("asprintf");
1528 3d1b16d1 2023-07-08 jrick goto done;
1529 3d1b16d1 2023-07-08 jrick }
1530 3d1b16d1 2023-07-08 jrick if (strcmp(path, "/") == 0) {
1531 3d1b16d1 2023-07-08 jrick parent_path = strdup("");
1532 3d1b16d1 2023-07-08 jrick if (parent_path == NULL)
1533 3d1b16d1 2023-07-08 jrick return got_error_from_errno("strdup");
1534 3d1b16d1 2023-07-08 jrick } else {
1535 3d1b16d1 2023-07-08 jrick err = got_path_dirname(&parent_path, path);
1536 3d1b16d1 2023-07-08 jrick if (err)
1537 3d1b16d1 2023-07-08 jrick return err;
1538 3d1b16d1 2023-07-08 jrick }
1539 3d1b16d1 2023-07-08 jrick
1540 3d1b16d1 2023-07-08 jrick ct = calloc(1, sizeof(*ct));
1541 3d1b16d1 2023-07-08 jrick if (ct == NULL) {
1542 3d1b16d1 2023-07-08 jrick err = got_error_from_errno("calloc");
1543 3d1b16d1 2023-07-08 jrick goto done;
1544 3d1b16d1 2023-07-08 jrick }
1545 3d1b16d1 2023-07-08 jrick
1546 3d1b16d1 2023-07-08 jrick if (asprintf(&ct->ondisk_path, "%s/%s", a->worktree->root_path,
1547 3d1b16d1 2023-07-08 jrick relpath) == -1) {
1548 3d1b16d1 2023-07-08 jrick err = got_error_from_errno("asprintf");
1549 3d1b16d1 2023-07-08 jrick goto done;
1550 3d1b16d1 2023-07-08 jrick }
1551 3d1b16d1 2023-07-08 jrick
1552 3d1b16d1 2023-07-08 jrick if (staged_status == GOT_STATUS_ADD ||
1553 3d1b16d1 2023-07-08 jrick staged_status == GOT_STATUS_MODIFY) {
1554 3d1b16d1 2023-07-08 jrick struct got_fileindex_entry *ie;
1555 3d1b16d1 2023-07-08 jrick ie = got_fileindex_entry_get(a->fileindex, path, strlen(path));
1556 3d1b16d1 2023-07-08 jrick switch (got_fileindex_entry_staged_filetype_get(ie)) {
1557 3d1b16d1 2023-07-08 jrick case GOT_FILEIDX_MODE_REGULAR_FILE:
1558 3d1b16d1 2023-07-08 jrick case GOT_FILEIDX_MODE_BAD_SYMLINK:
1559 3d1b16d1 2023-07-08 jrick ct->mode = S_IFREG;
1560 3d1b16d1 2023-07-08 jrick break;
1561 3d1b16d1 2023-07-08 jrick case GOT_FILEIDX_MODE_SYMLINK:
1562 3d1b16d1 2023-07-08 jrick ct->mode = S_IFLNK;
1563 3d1b16d1 2023-07-08 jrick break;
1564 3d1b16d1 2023-07-08 jrick default:
1565 3d1b16d1 2023-07-08 jrick err = got_error_path(path, GOT_ERR_BAD_FILETYPE);
1566 3d1b16d1 2023-07-08 jrick goto done;
1567 3d1b16d1 2023-07-08 jrick }
1568 3d1b16d1 2023-07-08 jrick ct->mode |= got_fileindex_entry_perms_get(ie);
1569 3d1b16d1 2023-07-08 jrick } else if (status != GOT_STATUS_DELETE &&
1570 3d1b16d1 2023-07-08 jrick staged_status != GOT_STATUS_DELETE) {
1571 3d1b16d1 2023-07-08 jrick if (dirfd != -1) {
1572 3d1b16d1 2023-07-08 jrick if (fstatat(dirfd, de_name, &sb,
1573 3d1b16d1 2023-07-08 jrick AT_SYMLINK_NOFOLLOW) == -1) {
1574 3d1b16d1 2023-07-08 jrick err = got_error_from_errno2("fstatat",
1575 3d1b16d1 2023-07-08 jrick ct->ondisk_path);
1576 3d1b16d1 2023-07-08 jrick goto done;
1577 3d1b16d1 2023-07-08 jrick }
1578 3d1b16d1 2023-07-08 jrick } else if (lstat(ct->ondisk_path, &sb) == -1) {
1579 3d1b16d1 2023-07-08 jrick err = got_error_from_errno2("lstat", ct->ondisk_path);
1580 3d1b16d1 2023-07-08 jrick goto done;
1581 3d1b16d1 2023-07-08 jrick }
1582 3d1b16d1 2023-07-08 jrick ct->mode = sb.st_mode;
1583 3d1b16d1 2023-07-08 jrick }
1584 3d1b16d1 2023-07-08 jrick
1585 3d1b16d1 2023-07-08 jrick if (asprintf(&ct->in_repo_path, "%s%s%s", a->worktree->path_prefix,
1586 3d1b16d1 2023-07-08 jrick got_path_is_root_dir(a->worktree->path_prefix) ? "" : "/",
1587 3d1b16d1 2023-07-08 jrick relpath) == -1) {
1588 3d1b16d1 2023-07-08 jrick err = got_error_from_errno("asprintf");
1589 3d1b16d1 2023-07-08 jrick goto done;
1590 3d1b16d1 2023-07-08 jrick }
1591 3d1b16d1 2023-07-08 jrick
1592 3d1b16d1 2023-07-08 jrick if (S_ISLNK(ct->mode) && staged_status == GOT_STATUS_NO_CHANGE &&
1593 3d1b16d1 2023-07-08 jrick status == GOT_STATUS_ADD && !a->allow_bad_symlinks) {
1594 3d1b16d1 2023-07-08 jrick int is_bad_symlink;
1595 3d1b16d1 2023-07-08 jrick char target_path[PATH_MAX];
1596 3d1b16d1 2023-07-08 jrick ssize_t target_len;
1597 3d1b16d1 2023-07-08 jrick target_len = readlink(ct->ondisk_path, target_path,
1598 3d1b16d1 2023-07-08 jrick sizeof(target_path));
1599 3d1b16d1 2023-07-08 jrick if (target_len == -1) {
1600 3d1b16d1 2023-07-08 jrick err = got_error_from_errno2("readlink",
1601 3d1b16d1 2023-07-08 jrick ct->ondisk_path);
1602 3d1b16d1 2023-07-08 jrick goto done;
1603 3d1b16d1 2023-07-08 jrick }
1604 3d1b16d1 2023-07-08 jrick err = is_bad_symlink_target(&is_bad_symlink, target_path,
1605 3d1b16d1 2023-07-08 jrick target_len, ct->ondisk_path, a->worktree->root_path);
1606 3d1b16d1 2023-07-08 jrick if (err)
1607 3d1b16d1 2023-07-08 jrick goto done;
1608 3d1b16d1 2023-07-08 jrick if (is_bad_symlink) {
1609 3d1b16d1 2023-07-08 jrick err = got_error_path(ct->ondisk_path,
1610 3d1b16d1 2023-07-08 jrick GOT_ERR_BAD_SYMLINK);
1611 3d1b16d1 2023-07-08 jrick goto done;
1612 3d1b16d1 2023-07-08 jrick }
1613 3d1b16d1 2023-07-08 jrick }
1614 3d1b16d1 2023-07-08 jrick
1615 3d1b16d1 2023-07-08 jrick ct->status = status;
1616 3d1b16d1 2023-07-08 jrick ct->staged_status = staged_status;
1617 3d1b16d1 2023-07-08 jrick ct->blob_id = NULL; /* will be filled in when blob gets created */
1618 3d1b16d1 2023-07-08 jrick if (ct->status != GOT_STATUS_ADD &&
1619 3d1b16d1 2023-07-08 jrick ct->staged_status != GOT_STATUS_ADD) {
1620 3d1b16d1 2023-07-08 jrick ct->base_blob_id = got_object_id_dup(blob_id);
1621 3d1b16d1 2023-07-08 jrick if (ct->base_blob_id == NULL) {
1622 3d1b16d1 2023-07-08 jrick err = got_error_from_errno("got_object_id_dup");
1623 3d1b16d1 2023-07-08 jrick goto done;
1624 3d1b16d1 2023-07-08 jrick }
1625 3d1b16d1 2023-07-08 jrick ct->base_commit_id = got_object_id_dup(commit_id);
1626 3d1b16d1 2023-07-08 jrick if (ct->base_commit_id == NULL) {
1627 3d1b16d1 2023-07-08 jrick err = got_error_from_errno("got_object_id_dup");
1628 3d1b16d1 2023-07-08 jrick goto done;
1629 3d1b16d1 2023-07-08 jrick }
1630 3d1b16d1 2023-07-08 jrick }
1631 3d1b16d1 2023-07-08 jrick if (ct->staged_status == GOT_STATUS_ADD ||
1632 3d1b16d1 2023-07-08 jrick ct->staged_status == GOT_STATUS_MODIFY) {
1633 3d1b16d1 2023-07-08 jrick ct->staged_blob_id = got_object_id_dup(staged_blob_id);
1634 3d1b16d1 2023-07-08 jrick if (ct->staged_blob_id == NULL) {
1635 3d1b16d1 2023-07-08 jrick err = got_error_from_errno("got_object_id_dup");
1636 3d1b16d1 2023-07-08 jrick goto done;
1637 3d1b16d1 2023-07-08 jrick }
1638 3d1b16d1 2023-07-08 jrick }
1639 3d1b16d1 2023-07-08 jrick ct->path = strdup(path);
1640 3d1b16d1 2023-07-08 jrick if (ct->path == NULL) {
1641 3d1b16d1 2023-07-08 jrick err = got_error_from_errno("strdup");
1642 3d1b16d1 2023-07-08 jrick goto done;
1643 3d1b16d1 2023-07-08 jrick }
1644 3d1b16d1 2023-07-08 jrick err = got_pathlist_insert(&new, a->commitable_paths, ct->path, ct);
1645 3d1b16d1 2023-07-08 jrick if (err)
1646 3d1b16d1 2023-07-08 jrick goto done;
1647 3d1b16d1 2023-07-08 jrick
1648 3d1b16d1 2023-07-08 jrick if (a->diff_outfile && ct && new != NULL) {
1649 3d1b16d1 2023-07-08 jrick err = append_ct_diff(ct, &a->diff_header_shown,
1650 3d1b16d1 2023-07-08 jrick a->diff_outfile, a->f1, a->f2, dirfd, de_name,
1651 3d1b16d1 2023-07-08 jrick a->have_staged_files, a->repo, a->worktree);
1652 3d1b16d1 2023-07-08 jrick if (err)
1653 3d1b16d1 2023-07-08 jrick goto done;
1654 3d1b16d1 2023-07-08 jrick }
1655 3d1b16d1 2023-07-08 jrick done:
1656 3d1b16d1 2023-07-08 jrick if (ct && (err || new == NULL))
1657 3d1b16d1 2023-07-08 jrick free_commitable(ct);
1658 3d1b16d1 2023-07-08 jrick free(parent_path);
1659 3d1b16d1 2023-07-08 jrick free(path);
1660 3d1b16d1 2023-07-08 jrick return err;
1661 3d1b16d1 2023-07-08 jrick }
1662 3d1b16d1 2023-07-08 jrick
1663 3d1b16d1 2023-07-08 jrick static const struct got_error *write_tree(struct got_object_id **, int *,
1664 3d1b16d1 2023-07-08 jrick struct got_tree_object *, const char *, struct got_pathlist_head *,
1665 3d1b16d1 2023-07-08 jrick got_worktree_status_cb status_cb, void *status_arg,
1666 3d1b16d1 2023-07-08 jrick struct got_repository *);
1667 3d1b16d1 2023-07-08 jrick
1668 3d1b16d1 2023-07-08 jrick static const struct got_error *
1669 3d1b16d1 2023-07-08 jrick write_subtree(struct got_object_id **new_subtree_id, int *nentries,
1670 3d1b16d1 2023-07-08 jrick struct got_tree_entry *te, const char *parent_path,
1671 3d1b16d1 2023-07-08 jrick struct got_pathlist_head *commitable_paths,
1672 3d1b16d1 2023-07-08 jrick got_worktree_status_cb status_cb, void *status_arg,
1673 3d1b16d1 2023-07-08 jrick struct got_repository *repo)
1674 3d1b16d1 2023-07-08 jrick {
1675 3d1b16d1 2023-07-08 jrick const struct got_error *err = NULL;
1676 3d1b16d1 2023-07-08 jrick struct got_tree_object *subtree;
1677 3d1b16d1 2023-07-08 jrick char *subpath;
1678 3d1b16d1 2023-07-08 jrick
1679 3d1b16d1 2023-07-08 jrick if (asprintf(&subpath, "%s%s%s", parent_path,
1680 3d1b16d1 2023-07-08 jrick got_path_is_root_dir(parent_path) ? "" : "/", te->name) == -1)
1681 3d1b16d1 2023-07-08 jrick return got_error_from_errno("asprintf");
1682 3d1b16d1 2023-07-08 jrick
1683 3d1b16d1 2023-07-08 jrick err = got_object_open_as_tree(&subtree, repo, &te->id);
1684 3d1b16d1 2023-07-08 jrick if (err)
1685 3d1b16d1 2023-07-08 jrick return err;
1686 3d1b16d1 2023-07-08 jrick
1687 3d1b16d1 2023-07-08 jrick err = write_tree(new_subtree_id, nentries, subtree, subpath,
1688 3d1b16d1 2023-07-08 jrick commitable_paths, status_cb, status_arg, repo);
1689 3d1b16d1 2023-07-08 jrick got_object_tree_close(subtree);
1690 3d1b16d1 2023-07-08 jrick free(subpath);
1691 3d1b16d1 2023-07-08 jrick return err;
1692 3d1b16d1 2023-07-08 jrick }
1693 3d1b16d1 2023-07-08 jrick
1694 3d1b16d1 2023-07-08 jrick static const struct got_error *
1695 3d1b16d1 2023-07-08 jrick match_ct_parent_path(int *match, struct got_commitable *ct, const char *path)
1696 3d1b16d1 2023-07-08 jrick {
1697 3d1b16d1 2023-07-08 jrick const struct got_error *err = NULL;
1698 3d1b16d1 2023-07-08 jrick char *ct_parent_path = NULL;
1699 3d1b16d1 2023-07-08 jrick
1700 3d1b16d1 2023-07-08 jrick *match = 0;
1701 3d1b16d1 2023-07-08 jrick
1702 3d1b16d1 2023-07-08 jrick if (strchr(ct->in_repo_path, '/') == NULL) {
1703 3d1b16d1 2023-07-08 jrick *match = got_path_is_root_dir(path);
1704 3d1b16d1 2023-07-08 jrick return NULL;
1705 3d1b16d1 2023-07-08 jrick }
1706 3d1b16d1 2023-07-08 jrick
1707 3d1b16d1 2023-07-08 jrick err = got_path_dirname(&ct_parent_path, ct->in_repo_path);
1708 3d1b16d1 2023-07-08 jrick if (err)
1709 3d1b16d1 2023-07-08 jrick return err;
1710 3d1b16d1 2023-07-08 jrick *match = (strcmp(path, ct_parent_path) == 0);
1711 3d1b16d1 2023-07-08 jrick free(ct_parent_path);
1712 3d1b16d1 2023-07-08 jrick return err;
1713 3d1b16d1 2023-07-08 jrick }
1714 3d1b16d1 2023-07-08 jrick
1715 3d1b16d1 2023-07-08 jrick static mode_t
1716 3d1b16d1 2023-07-08 jrick get_ct_file_mode(struct got_commitable *ct)
1717 3d1b16d1 2023-07-08 jrick {
1718 3d1b16d1 2023-07-08 jrick if (S_ISLNK(ct->mode))
1719 3d1b16d1 2023-07-08 jrick return S_IFLNK;
1720 3d1b16d1 2023-07-08 jrick
1721 3d1b16d1 2023-07-08 jrick return S_IFREG | (ct->mode & ((S_IRWXU | S_IRWXG | S_IRWXO)));
1722 3d1b16d1 2023-07-08 jrick }
1723 3d1b16d1 2023-07-08 jrick
1724 3d1b16d1 2023-07-08 jrick static const struct got_error *
1725 3d1b16d1 2023-07-08 jrick alloc_modified_blob_tree_entry(struct got_tree_entry **new_te,
1726 3d1b16d1 2023-07-08 jrick struct got_tree_entry *te, struct got_commitable *ct)
1727 3d1b16d1 2023-07-08 jrick {
1728 3d1b16d1 2023-07-08 jrick const struct got_error *err = NULL;
1729 3d1b16d1 2023-07-08 jrick
1730 3d1b16d1 2023-07-08 jrick *new_te = NULL;
1731 3d1b16d1 2023-07-08 jrick
1732 3d1b16d1 2023-07-08 jrick err = got_object_tree_entry_dup(new_te, te);
1733 3d1b16d1 2023-07-08 jrick if (err)
1734 3d1b16d1 2023-07-08 jrick goto done;
1735 3d1b16d1 2023-07-08 jrick
1736 3d1b16d1 2023-07-08 jrick (*new_te)->mode = get_ct_file_mode(ct);
1737 3d1b16d1 2023-07-08 jrick
1738 3d1b16d1 2023-07-08 jrick if (ct->staged_status == GOT_STATUS_MODIFY)
1739 3d1b16d1 2023-07-08 jrick memcpy(&(*new_te)->id, ct->staged_blob_id,
1740 3d1b16d1 2023-07-08 jrick sizeof((*new_te)->id));
1741 3d1b16d1 2023-07-08 jrick else
1742 3d1b16d1 2023-07-08 jrick memcpy(&(*new_te)->id, ct->blob_id, sizeof((*new_te)->id));
1743 3d1b16d1 2023-07-08 jrick done:
1744 3d1b16d1 2023-07-08 jrick if (err && *new_te) {
1745 3d1b16d1 2023-07-08 jrick free(*new_te);
1746 3d1b16d1 2023-07-08 jrick *new_te = NULL;
1747 3d1b16d1 2023-07-08 jrick }
1748 3d1b16d1 2023-07-08 jrick return err;
1749 3d1b16d1 2023-07-08 jrick }
1750 3d1b16d1 2023-07-08 jrick
1751 3d1b16d1 2023-07-08 jrick static const struct got_error *
1752 3d1b16d1 2023-07-08 jrick alloc_added_blob_tree_entry(struct got_tree_entry **new_te,
1753 3d1b16d1 2023-07-08 jrick struct got_commitable *ct)
1754 3d1b16d1 2023-07-08 jrick {
1755 3d1b16d1 2023-07-08 jrick const struct got_error *err = NULL;
1756 3d1b16d1 2023-07-08 jrick char *ct_name = NULL;
1757 3d1b16d1 2023-07-08 jrick
1758 3d1b16d1 2023-07-08 jrick *new_te = NULL;
1759 3d1b16d1 2023-07-08 jrick
1760 3d1b16d1 2023-07-08 jrick *new_te = calloc(1, sizeof(**new_te));
1761 3d1b16d1 2023-07-08 jrick if (*new_te == NULL)
1762 3d1b16d1 2023-07-08 jrick return got_error_from_errno("calloc");
1763 3d1b16d1 2023-07-08 jrick
1764 3d1b16d1 2023-07-08 jrick err = got_path_basename(&ct_name, ct->path);
1765 3d1b16d1 2023-07-08 jrick if (err)
1766 3d1b16d1 2023-07-08 jrick goto done;
1767 3d1b16d1 2023-07-08 jrick if (strlcpy((*new_te)->name, ct_name, sizeof((*new_te)->name)) >=
1768 3d1b16d1 2023-07-08 jrick sizeof((*new_te)->name)) {
1769 3d1b16d1 2023-07-08 jrick err = got_error(GOT_ERR_NO_SPACE);
1770 3d1b16d1 2023-07-08 jrick goto done;
1771 3d1b16d1 2023-07-08 jrick }
1772 3d1b16d1 2023-07-08 jrick
1773 3d1b16d1 2023-07-08 jrick (*new_te)->mode = get_ct_file_mode(ct);
1774 3d1b16d1 2023-07-08 jrick
1775 3d1b16d1 2023-07-08 jrick if (ct->staged_status == GOT_STATUS_ADD)
1776 3d1b16d1 2023-07-08 jrick memcpy(&(*new_te)->id, ct->staged_blob_id,
1777 3d1b16d1 2023-07-08 jrick sizeof((*new_te)->id));
1778 3d1b16d1 2023-07-08 jrick else
1779 3d1b16d1 2023-07-08 jrick memcpy(&(*new_te)->id, ct->blob_id, sizeof((*new_te)->id));
1780 3d1b16d1 2023-07-08 jrick done:
1781 3d1b16d1 2023-07-08 jrick free(ct_name);
1782 3d1b16d1 2023-07-08 jrick if (err && *new_te) {
1783 3d1b16d1 2023-07-08 jrick free(*new_te);
1784 3d1b16d1 2023-07-08 jrick *new_te = NULL;
1785 3d1b16d1 2023-07-08 jrick }
1786 3d1b16d1 2023-07-08 jrick return err;
1787 3d1b16d1 2023-07-08 jrick }
1788 3d1b16d1 2023-07-08 jrick
1789 3d1b16d1 2023-07-08 jrick static const struct got_error *
1790 3d1b16d1 2023-07-08 jrick insert_tree_entry(struct got_tree_entry *new_te,
1791 3d1b16d1 2023-07-08 jrick struct got_pathlist_head *paths)
1792 3d1b16d1 2023-07-08 jrick {
1793 3d1b16d1 2023-07-08 jrick const struct got_error *err = NULL;
1794 3d1b16d1 2023-07-08 jrick struct got_pathlist_entry *new_pe;
1795 3d1b16d1 2023-07-08 jrick
1796 3d1b16d1 2023-07-08 jrick err = got_pathlist_insert(&new_pe, paths, new_te->name, new_te);
1797 3d1b16d1 2023-07-08 jrick if (err)
1798 3d1b16d1 2023-07-08 jrick return err;
1799 3d1b16d1 2023-07-08 jrick if (new_pe == NULL)
1800 3d1b16d1 2023-07-08 jrick return got_error(GOT_ERR_TREE_DUP_ENTRY);
1801 3d1b16d1 2023-07-08 jrick return NULL;
1802 3d1b16d1 2023-07-08 jrick }
1803 3d1b16d1 2023-07-08 jrick
1804 3d1b16d1 2023-07-08 jrick static const struct got_error *
1805 3d1b16d1 2023-07-08 jrick report_ct_status(struct got_commitable *ct,
1806 3d1b16d1 2023-07-08 jrick got_worktree_status_cb status_cb, void *status_arg)
1807 3d1b16d1 2023-07-08 jrick {
1808 3d1b16d1 2023-07-08 jrick const char *ct_path = ct->path;
1809 3d1b16d1 2023-07-08 jrick unsigned char status;
1810 3d1b16d1 2023-07-08 jrick
1811 3d1b16d1 2023-07-08 jrick if (status_cb == NULL) /* no commit progress output desired */
1812 3d1b16d1 2023-07-08 jrick return NULL;
1813 3d1b16d1 2023-07-08 jrick
1814 3d1b16d1 2023-07-08 jrick while (ct_path[0] == '/')
1815 3d1b16d1 2023-07-08 jrick ct_path++;
1816 3d1b16d1 2023-07-08 jrick
1817 3d1b16d1 2023-07-08 jrick if (ct->staged_status != GOT_STATUS_NO_CHANGE)
1818 3d1b16d1 2023-07-08 jrick status = ct->staged_status;
1819 3d1b16d1 2023-07-08 jrick else
1820 3d1b16d1 2023-07-08 jrick status = ct->status;
1821 3d1b16d1 2023-07-08 jrick
1822 3d1b16d1 2023-07-08 jrick return (*status_cb)(status_arg, status, GOT_STATUS_NO_CHANGE,
1823 3d1b16d1 2023-07-08 jrick ct_path, ct->blob_id, NULL, NULL, -1, NULL);
1824 3d1b16d1 2023-07-08 jrick }
1825 3d1b16d1 2023-07-08 jrick
1826 3d1b16d1 2023-07-08 jrick static const struct got_error *
1827 3d1b16d1 2023-07-08 jrick match_modified_subtree(int *modified, struct got_tree_entry *te,
1828 3d1b16d1 2023-07-08 jrick const char *base_tree_path, struct got_pathlist_head *commitable_paths)
1829 3d1b16d1 2023-07-08 jrick {
1830 3d1b16d1 2023-07-08 jrick const struct got_error *err = NULL;
1831 3d1b16d1 2023-07-08 jrick struct got_pathlist_entry *pe;
1832 3d1b16d1 2023-07-08 jrick char *te_path;
1833 3d1b16d1 2023-07-08 jrick
1834 3d1b16d1 2023-07-08 jrick *modified = 0;
1835 3d1b16d1 2023-07-08 jrick
1836 3d1b16d1 2023-07-08 jrick if (asprintf(&te_path, "%s%s%s", base_tree_path,
1837 3d1b16d1 2023-07-08 jrick got_path_is_root_dir(base_tree_path) ? "" : "/",
1838 3d1b16d1 2023-07-08 jrick te->name) == -1)
1839 3d1b16d1 2023-07-08 jrick return got_error_from_errno("asprintf");
1840 3d1b16d1 2023-07-08 jrick
1841 3d1b16d1 2023-07-08 jrick TAILQ_FOREACH(pe, commitable_paths, entry) {
1842 3d1b16d1 2023-07-08 jrick struct got_commitable *ct = pe->data;
1843 3d1b16d1 2023-07-08 jrick *modified = got_path_is_child(ct->in_repo_path, te_path,
1844 3d1b16d1 2023-07-08 jrick strlen(te_path));
1845 3d1b16d1 2023-07-08 jrick if (*modified)
1846 3d1b16d1 2023-07-08 jrick break;
1847 3d1b16d1 2023-07-08 jrick }
1848 3d1b16d1 2023-07-08 jrick
1849 3d1b16d1 2023-07-08 jrick free(te_path);
1850 3d1b16d1 2023-07-08 jrick return err;
1851 3d1b16d1 2023-07-08 jrick }
1852 3d1b16d1 2023-07-08 jrick
1853 3d1b16d1 2023-07-08 jrick static const struct got_error *
1854 3d1b16d1 2023-07-08 jrick match_deleted_or_modified_ct(struct got_commitable **ctp,
1855 3d1b16d1 2023-07-08 jrick struct got_tree_entry *te, const char *base_tree_path,
1856 3d1b16d1 2023-07-08 jrick struct got_pathlist_head *commitable_paths)
1857 3d1b16d1 2023-07-08 jrick {
1858 3d1b16d1 2023-07-08 jrick const struct got_error *err = NULL;
1859 3d1b16d1 2023-07-08 jrick struct got_pathlist_entry *pe;
1860 3d1b16d1 2023-07-08 jrick
1861 3d1b16d1 2023-07-08 jrick *ctp = NULL;
1862 3d1b16d1 2023-07-08 jrick
1863 3d1b16d1 2023-07-08 jrick TAILQ_FOREACH(pe, commitable_paths, entry) {
1864 3d1b16d1 2023-07-08 jrick struct got_commitable *ct = pe->data;
1865 3d1b16d1 2023-07-08 jrick char *ct_name = NULL;
1866 3d1b16d1 2023-07-08 jrick int path_matches;
1867 3d1b16d1 2023-07-08 jrick
1868 3d1b16d1 2023-07-08 jrick if (ct->staged_status == GOT_STATUS_NO_CHANGE) {
1869 3d1b16d1 2023-07-08 jrick if (ct->status != GOT_STATUS_MODIFY &&
1870 3d1b16d1 2023-07-08 jrick ct->status != GOT_STATUS_MODE_CHANGE &&
1871 3d1b16d1 2023-07-08 jrick ct->status != GOT_STATUS_DELETE &&
1872 3d1b16d1 2023-07-08 jrick ct->status != GOT_STATUS_CONFLICT)
1873 3d1b16d1 2023-07-08 jrick continue;
1874 3d1b16d1 2023-07-08 jrick } else {
1875 3d1b16d1 2023-07-08 jrick if (ct->staged_status != GOT_STATUS_MODIFY &&
1876 3d1b16d1 2023-07-08 jrick ct->staged_status != GOT_STATUS_DELETE)
1877 3d1b16d1 2023-07-08 jrick continue;
1878 3d1b16d1 2023-07-08 jrick }
1879 3d1b16d1 2023-07-08 jrick
1880 3d1b16d1 2023-07-08 jrick if (got_object_id_cmp(ct->base_blob_id, &te->id) != 0)
1881 3d1b16d1 2023-07-08 jrick continue;
1882 3d1b16d1 2023-07-08 jrick
1883 3d1b16d1 2023-07-08 jrick err = match_ct_parent_path(&path_matches, ct, base_tree_path);
1884 3d1b16d1 2023-07-08 jrick if (err)
1885 3d1b16d1 2023-07-08 jrick return err;
1886 3d1b16d1 2023-07-08 jrick if (!path_matches)
1887 3d1b16d1 2023-07-08 jrick continue;
1888 3d1b16d1 2023-07-08 jrick
1889 3d1b16d1 2023-07-08 jrick err = got_path_basename(&ct_name, pe->path);
1890 3d1b16d1 2023-07-08 jrick if (err)
1891 3d1b16d1 2023-07-08 jrick return err;
1892 3d1b16d1 2023-07-08 jrick
1893 3d1b16d1 2023-07-08 jrick if (strcmp(te->name, ct_name) != 0) {
1894 3d1b16d1 2023-07-08 jrick free(ct_name);
1895 3d1b16d1 2023-07-08 jrick continue;
1896 3d1b16d1 2023-07-08 jrick }
1897 3d1b16d1 2023-07-08 jrick free(ct_name);
1898 3d1b16d1 2023-07-08 jrick
1899 3d1b16d1 2023-07-08 jrick *ctp = ct;
1900 3d1b16d1 2023-07-08 jrick break;
1901 3d1b16d1 2023-07-08 jrick }
1902 3d1b16d1 2023-07-08 jrick
1903 3d1b16d1 2023-07-08 jrick return err;
1904 3d1b16d1 2023-07-08 jrick }
1905 3d1b16d1 2023-07-08 jrick
1906 3d1b16d1 2023-07-08 jrick static const struct got_error *
1907 3d1b16d1 2023-07-08 jrick make_subtree_for_added_blob(struct got_tree_entry **new_tep,
1908 3d1b16d1 2023-07-08 jrick const char *child_path, const char *path_base_tree,
1909 3d1b16d1 2023-07-08 jrick struct got_pathlist_head *commitable_paths,
1910 3d1b16d1 2023-07-08 jrick got_worktree_status_cb status_cb, void *status_arg,
1911 3d1b16d1 2023-07-08 jrick struct got_repository *repo)
1912 3d1b16d1 2023-07-08 jrick {
1913 3d1b16d1 2023-07-08 jrick const struct got_error *err = NULL;
1914 3d1b16d1 2023-07-08 jrick struct got_tree_entry *new_te;
1915 3d1b16d1 2023-07-08 jrick char *subtree_path;
1916 3d1b16d1 2023-07-08 jrick struct got_object_id *id = NULL;
1917 3d1b16d1 2023-07-08 jrick int nentries;
1918 3d1b16d1 2023-07-08 jrick
1919 3d1b16d1 2023-07-08 jrick *new_tep = NULL;
1920 3d1b16d1 2023-07-08 jrick
1921 3d1b16d1 2023-07-08 jrick if (asprintf(&subtree_path, "%s%s%s", path_base_tree,
1922 3d1b16d1 2023-07-08 jrick got_path_is_root_dir(path_base_tree) ? "" : "/",
1923 3d1b16d1 2023-07-08 jrick child_path) == -1)
1924 3d1b16d1 2023-07-08 jrick return got_error_from_errno("asprintf");
1925 3d1b16d1 2023-07-08 jrick
1926 3d1b16d1 2023-07-08 jrick new_te = calloc(1, sizeof(*new_te));
1927 3d1b16d1 2023-07-08 jrick if (new_te == NULL)
1928 3d1b16d1 2023-07-08 jrick return got_error_from_errno("calloc");
1929 3d1b16d1 2023-07-08 jrick new_te->mode = S_IFDIR;
1930 3d1b16d1 2023-07-08 jrick
1931 3d1b16d1 2023-07-08 jrick if (strlcpy(new_te->name, child_path, sizeof(new_te->name)) >=
1932 3d1b16d1 2023-07-08 jrick sizeof(new_te->name)) {
1933 3d1b16d1 2023-07-08 jrick err = got_error(GOT_ERR_NO_SPACE);
1934 3d1b16d1 2023-07-08 jrick goto done;
1935 3d1b16d1 2023-07-08 jrick }
1936 3d1b16d1 2023-07-08 jrick err = write_tree(&id, &nentries, NULL, subtree_path,
1937 3d1b16d1 2023-07-08 jrick commitable_paths, status_cb, status_arg, repo);
1938 3d1b16d1 2023-07-08 jrick if (err) {
1939 3d1b16d1 2023-07-08 jrick free(new_te);
1940 3d1b16d1 2023-07-08 jrick goto done;
1941 3d1b16d1 2023-07-08 jrick }
1942 3d1b16d1 2023-07-08 jrick memcpy(&new_te->id, id, sizeof(new_te->id));
1943 3d1b16d1 2023-07-08 jrick done:
1944 3d1b16d1 2023-07-08 jrick free(id);
1945 3d1b16d1 2023-07-08 jrick free(subtree_path);
1946 3d1b16d1 2023-07-08 jrick if (err == NULL)
1947 3d1b16d1 2023-07-08 jrick *new_tep = new_te;
1948 3d1b16d1 2023-07-08 jrick return err;
1949 3d1b16d1 2023-07-08 jrick }
1950 3d1b16d1 2023-07-08 jrick
1951 3d1b16d1 2023-07-08 jrick static const struct got_error *
1952 3d1b16d1 2023-07-08 jrick write_tree(struct got_object_id **new_tree_id, int *nentries,
1953 3d1b16d1 2023-07-08 jrick struct got_tree_object *base_tree, const char *path_base_tree,
1954 3d1b16d1 2023-07-08 jrick struct got_pathlist_head *commitable_paths,
1955 3d1b16d1 2023-07-08 jrick got_worktree_status_cb status_cb, void *status_arg,
1956 3d1b16d1 2023-07-08 jrick struct got_repository *repo)
1957 3d1b16d1 2023-07-08 jrick {
1958 3d1b16d1 2023-07-08 jrick const struct got_error *err = NULL;
1959 3d1b16d1 2023-07-08 jrick struct got_pathlist_head paths;
1960 3d1b16d1 2023-07-08 jrick struct got_tree_entry *te, *new_te = NULL;
1961 3d1b16d1 2023-07-08 jrick struct got_pathlist_entry *pe;
1962 3d1b16d1 2023-07-08 jrick
1963 3d1b16d1 2023-07-08 jrick TAILQ_INIT(&paths);
1964 3d1b16d1 2023-07-08 jrick *nentries = 0;
1965 3d1b16d1 2023-07-08 jrick
1966 3d1b16d1 2023-07-08 jrick /* Insert, and recurse into, newly added entries first. */
1967 3d1b16d1 2023-07-08 jrick TAILQ_FOREACH(pe, commitable_paths, entry) {
1968 3d1b16d1 2023-07-08 jrick struct got_commitable *ct = pe->data;
1969 3d1b16d1 2023-07-08 jrick char *child_path = NULL, *slash;
1970 3d1b16d1 2023-07-08 jrick
1971 3d1b16d1 2023-07-08 jrick if ((ct->status != GOT_STATUS_ADD &&
1972 3d1b16d1 2023-07-08 jrick ct->staged_status != GOT_STATUS_ADD) ||
1973 3d1b16d1 2023-07-08 jrick (ct->flags & GOT_COMMITABLE_ADDED))
1974 3d1b16d1 2023-07-08 jrick continue;
1975 3d1b16d1 2023-07-08 jrick
1976 3d1b16d1 2023-07-08 jrick if (!got_path_is_child(ct->in_repo_path, path_base_tree,
1977 3d1b16d1 2023-07-08 jrick strlen(path_base_tree)))
1978 3d1b16d1 2023-07-08 jrick continue;
1979 3d1b16d1 2023-07-08 jrick
1980 3d1b16d1 2023-07-08 jrick err = got_path_skip_common_ancestor(&child_path, path_base_tree,
1981 3d1b16d1 2023-07-08 jrick ct->in_repo_path);
1982 3d1b16d1 2023-07-08 jrick if (err)
1983 3d1b16d1 2023-07-08 jrick goto done;
1984 3d1b16d1 2023-07-08 jrick
1985 3d1b16d1 2023-07-08 jrick slash = strchr(child_path, '/');
1986 3d1b16d1 2023-07-08 jrick if (slash == NULL) {
1987 3d1b16d1 2023-07-08 jrick err = alloc_added_blob_tree_entry(&new_te, ct);
1988 3d1b16d1 2023-07-08 jrick if (err)
1989 3d1b16d1 2023-07-08 jrick goto done;
1990 3d1b16d1 2023-07-08 jrick err = report_ct_status(ct, status_cb, status_arg);
1991 3d1b16d1 2023-07-08 jrick if (err)
1992 3d1b16d1 2023-07-08 jrick goto done;
1993 3d1b16d1 2023-07-08 jrick ct->flags |= GOT_COMMITABLE_ADDED;
1994 3d1b16d1 2023-07-08 jrick err = insert_tree_entry(new_te, &paths);
1995 3d1b16d1 2023-07-08 jrick if (err)
1996 3d1b16d1 2023-07-08 jrick goto done;
1997 3d1b16d1 2023-07-08 jrick (*nentries)++;
1998 3d1b16d1 2023-07-08 jrick } else {
1999 3d1b16d1 2023-07-08 jrick *slash = '\0'; /* trim trailing path components */
2000 3d1b16d1 2023-07-08 jrick if (base_tree == NULL ||
2001 3d1b16d1 2023-07-08 jrick got_object_tree_find_entry(base_tree, child_path)
2002 3d1b16d1 2023-07-08 jrick == NULL) {
2003 3d1b16d1 2023-07-08 jrick err = make_subtree_for_added_blob(&new_te,
2004 3d1b16d1 2023-07-08 jrick child_path, path_base_tree,
2005 3d1b16d1 2023-07-08 jrick commitable_paths, status_cb, status_arg,
2006 3d1b16d1 2023-07-08 jrick repo);
2007 3d1b16d1 2023-07-08 jrick if (err)
2008 3d1b16d1 2023-07-08 jrick goto done;
2009 3d1b16d1 2023-07-08 jrick err = insert_tree_entry(new_te, &paths);
2010 3d1b16d1 2023-07-08 jrick if (err)
2011 3d1b16d1 2023-07-08 jrick goto done;
2012 3d1b16d1 2023-07-08 jrick (*nentries)++;
2013 3d1b16d1 2023-07-08 jrick }
2014 3d1b16d1 2023-07-08 jrick }
2015 3d1b16d1 2023-07-08 jrick }
2016 3d1b16d1 2023-07-08 jrick
2017 3d1b16d1 2023-07-08 jrick if (base_tree) {
2018 3d1b16d1 2023-07-08 jrick int i, nbase_entries;
2019 3d1b16d1 2023-07-08 jrick /* Handle modified and deleted entries. */
2020 3d1b16d1 2023-07-08 jrick nbase_entries = got_object_tree_get_nentries(base_tree);
2021 3d1b16d1 2023-07-08 jrick for (i = 0; i < nbase_entries; i++) {
2022 3d1b16d1 2023-07-08 jrick struct got_commitable *ct = NULL;
2023 3d1b16d1 2023-07-08 jrick
2024 3d1b16d1 2023-07-08 jrick te = got_object_tree_get_entry(base_tree, i);
2025 3d1b16d1 2023-07-08 jrick if (got_object_tree_entry_is_submodule(te)) {
2026 3d1b16d1 2023-07-08 jrick /* Entry is a submodule; just copy it. */
2027 3d1b16d1 2023-07-08 jrick err = got_object_tree_entry_dup(&new_te, te);
2028 3d1b16d1 2023-07-08 jrick if (err)
2029 3d1b16d1 2023-07-08 jrick goto done;
2030 3d1b16d1 2023-07-08 jrick err = insert_tree_entry(new_te, &paths);
2031 3d1b16d1 2023-07-08 jrick if (err)
2032 3d1b16d1 2023-07-08 jrick goto done;
2033 3d1b16d1 2023-07-08 jrick (*nentries)++;
2034 3d1b16d1 2023-07-08 jrick continue;
2035 3d1b16d1 2023-07-08 jrick }
2036 3d1b16d1 2023-07-08 jrick
2037 3d1b16d1 2023-07-08 jrick if (S_ISDIR(te->mode)) {
2038 3d1b16d1 2023-07-08 jrick int modified;
2039 3d1b16d1 2023-07-08 jrick err = got_object_tree_entry_dup(&new_te, te);
2040 3d1b16d1 2023-07-08 jrick if (err)
2041 3d1b16d1 2023-07-08 jrick goto done;
2042 3d1b16d1 2023-07-08 jrick err = match_modified_subtree(&modified, te,
2043 3d1b16d1 2023-07-08 jrick path_base_tree, commitable_paths);
2044 3d1b16d1 2023-07-08 jrick if (err)
2045 3d1b16d1 2023-07-08 jrick goto done;
2046 3d1b16d1 2023-07-08 jrick /* Avoid recursion into unmodified subtrees. */
2047 3d1b16d1 2023-07-08 jrick if (modified) {
2048 3d1b16d1 2023-07-08 jrick struct got_object_id *new_id;
2049 3d1b16d1 2023-07-08 jrick int nsubentries;
2050 3d1b16d1 2023-07-08 jrick err = write_subtree(&new_id,
2051 3d1b16d1 2023-07-08 jrick &nsubentries, te,
2052 3d1b16d1 2023-07-08 jrick path_base_tree, commitable_paths,
2053 3d1b16d1 2023-07-08 jrick status_cb, status_arg, repo);
2054 3d1b16d1 2023-07-08 jrick if (err)
2055 3d1b16d1 2023-07-08 jrick goto done;
2056 3d1b16d1 2023-07-08 jrick if (nsubentries == 0) {
2057 3d1b16d1 2023-07-08 jrick /* All entries were deleted. */
2058 3d1b16d1 2023-07-08 jrick free(new_id);
2059 3d1b16d1 2023-07-08 jrick continue;
2060 3d1b16d1 2023-07-08 jrick }
2061 3d1b16d1 2023-07-08 jrick memcpy(&new_te->id, new_id,
2062 3d1b16d1 2023-07-08 jrick sizeof(new_te->id));
2063 3d1b16d1 2023-07-08 jrick free(new_id);
2064 3d1b16d1 2023-07-08 jrick }
2065 3d1b16d1 2023-07-08 jrick err = insert_tree_entry(new_te, &paths);
2066 3d1b16d1 2023-07-08 jrick if (err)
2067 3d1b16d1 2023-07-08 jrick goto done;
2068 3d1b16d1 2023-07-08 jrick (*nentries)++;
2069 3d1b16d1 2023-07-08 jrick continue;
2070 3d1b16d1 2023-07-08 jrick }
2071 3d1b16d1 2023-07-08 jrick
2072 3d1b16d1 2023-07-08 jrick err = match_deleted_or_modified_ct(&ct, te,
2073 3d1b16d1 2023-07-08 jrick path_base_tree, commitable_paths);
2074 3d1b16d1 2023-07-08 jrick if (err)
2075 3d1b16d1 2023-07-08 jrick goto done;
2076 3d1b16d1 2023-07-08 jrick if (ct) {
2077 3d1b16d1 2023-07-08 jrick /* NB: Deleted entries get dropped here. */
2078 3d1b16d1 2023-07-08 jrick if (ct->status == GOT_STATUS_MODIFY ||
2079 3d1b16d1 2023-07-08 jrick ct->status == GOT_STATUS_MODE_CHANGE ||
2080 3d1b16d1 2023-07-08 jrick ct->status == GOT_STATUS_CONFLICT ||
2081 3d1b16d1 2023-07-08 jrick ct->staged_status == GOT_STATUS_MODIFY) {
2082 3d1b16d1 2023-07-08 jrick err = alloc_modified_blob_tree_entry(
2083 3d1b16d1 2023-07-08 jrick &new_te, te, ct);
2084 3d1b16d1 2023-07-08 jrick if (err)
2085 3d1b16d1 2023-07-08 jrick goto done;
2086 3d1b16d1 2023-07-08 jrick err = insert_tree_entry(new_te, &paths);
2087 3d1b16d1 2023-07-08 jrick if (err)
2088 3d1b16d1 2023-07-08 jrick goto done;
2089 3d1b16d1 2023-07-08 jrick (*nentries)++;
2090 3d1b16d1 2023-07-08 jrick }
2091 3d1b16d1 2023-07-08 jrick err = report_ct_status(ct, status_cb,
2092 3d1b16d1 2023-07-08 jrick status_arg);
2093 3d1b16d1 2023-07-08 jrick if (err)
2094 3d1b16d1 2023-07-08 jrick goto done;
2095 3d1b16d1 2023-07-08 jrick } else {
2096 3d1b16d1 2023-07-08 jrick /* Entry is unchanged; just copy it. */
2097 3d1b16d1 2023-07-08 jrick err = got_object_tree_entry_dup(&new_te, te);
2098 3d1b16d1 2023-07-08 jrick if (err)
2099 3d1b16d1 2023-07-08 jrick goto done;
2100 3d1b16d1 2023-07-08 jrick err = insert_tree_entry(new_te, &paths);
2101 3d1b16d1 2023-07-08 jrick if (err)
2102 3d1b16d1 2023-07-08 jrick goto done;
2103 3d1b16d1 2023-07-08 jrick (*nentries)++;
2104 3d1b16d1 2023-07-08 jrick }
2105 3d1b16d1 2023-07-08 jrick }
2106 3d1b16d1 2023-07-08 jrick }
2107 3d1b16d1 2023-07-08 jrick
2108 3d1b16d1 2023-07-08 jrick /* Write new list of entries; deleted entries have been dropped. */
2109 3d1b16d1 2023-07-08 jrick err = got_object_tree_create(new_tree_id, &paths, *nentries, repo);
2110 3d1b16d1 2023-07-08 jrick done:
2111 3d1b16d1 2023-07-08 jrick got_pathlist_free(&paths, GOT_PATHLIST_FREE_NONE);
2112 3d1b16d1 2023-07-08 jrick return err;
2113 3d1b16d1 2023-07-08 jrick }
2114 3d1b16d1 2023-07-08 jrick
2115 3d1b16d1 2023-07-08 jrick static const struct got_error *
2116 3d1b16d1 2023-07-08 jrick update_fileindex_after_commit(struct got_worktree *worktree,
2117 3d1b16d1 2023-07-08 jrick struct got_pathlist_head *commitable_paths,
2118 3d1b16d1 2023-07-08 jrick struct got_object_id *new_base_commit_id,
2119 3d1b16d1 2023-07-08 jrick struct got_fileindex *fileindex, int have_staged_files)
2120 3d1b16d1 2023-07-08 jrick {
2121 3d1b16d1 2023-07-08 jrick const struct got_error *err = NULL;
2122 3d1b16d1 2023-07-08 jrick struct got_pathlist_entry *pe;
2123 3d1b16d1 2023-07-08 jrick char *relpath = NULL;
2124 3d1b16d1 2023-07-08 jrick
2125 3d1b16d1 2023-07-08 jrick TAILQ_FOREACH(pe, commitable_paths, entry) {
2126 3d1b16d1 2023-07-08 jrick struct got_fileindex_entry *ie;
2127 3d1b16d1 2023-07-08 jrick struct got_commitable *ct = pe->data;
2128 3d1b16d1 2023-07-08 jrick
2129 3d1b16d1 2023-07-08 jrick ie = got_fileindex_entry_get(fileindex, pe->path, pe->path_len);
2130 3d1b16d1 2023-07-08 jrick
2131 3d1b16d1 2023-07-08 jrick err = got_path_skip_common_ancestor(&relpath,
2132 3d1b16d1 2023-07-08 jrick worktree->root_path, ct->ondisk_path);
2133 3d1b16d1 2023-07-08 jrick if (err)
2134 3d1b16d1 2023-07-08 jrick goto done;
2135 3d1b16d1 2023-07-08 jrick
2136 3d1b16d1 2023-07-08 jrick if (ie) {
2137 3d1b16d1 2023-07-08 jrick if (ct->status == GOT_STATUS_DELETE ||
2138 3d1b16d1 2023-07-08 jrick ct->staged_status == GOT_STATUS_DELETE) {
2139 3d1b16d1 2023-07-08 jrick got_fileindex_entry_remove(fileindex, ie);
2140 3d1b16d1 2023-07-08 jrick } else if (ct->staged_status == GOT_STATUS_ADD ||
2141 3d1b16d1 2023-07-08 jrick ct->staged_status == GOT_STATUS_MODIFY) {
2142 3d1b16d1 2023-07-08 jrick got_fileindex_entry_stage_set(ie,
2143 3d1b16d1 2023-07-08 jrick GOT_FILEIDX_STAGE_NONE);
2144 3d1b16d1 2023-07-08 jrick got_fileindex_entry_staged_filetype_set(ie, 0);
2145 3d1b16d1 2023-07-08 jrick
2146 3d1b16d1 2023-07-08 jrick err = got_fileindex_entry_update(ie,
2147 3d1b16d1 2023-07-08 jrick worktree->root_fd, relpath,
2148 3d1b16d1 2023-07-08 jrick ct->staged_blob_id->sha1,
2149 3d1b16d1 2023-07-08 jrick new_base_commit_id->sha1,
2150 3d1b16d1 2023-07-08 jrick !have_staged_files);
2151 3d1b16d1 2023-07-08 jrick } else
2152 3d1b16d1 2023-07-08 jrick err = got_fileindex_entry_update(ie,
2153 3d1b16d1 2023-07-08 jrick worktree->root_fd, relpath,
2154 3d1b16d1 2023-07-08 jrick ct->blob_id->sha1,
2155 3d1b16d1 2023-07-08 jrick new_base_commit_id->sha1,
2156 3d1b16d1 2023-07-08 jrick !have_staged_files);
2157 3d1b16d1 2023-07-08 jrick } else {
2158 3d1b16d1 2023-07-08 jrick err = got_fileindex_entry_alloc(&ie, pe->path);
2159 3d1b16d1 2023-07-08 jrick if (err)
2160 3d1b16d1 2023-07-08 jrick goto done;
2161 3d1b16d1 2023-07-08 jrick err = got_fileindex_entry_update(ie,
2162 3d1b16d1 2023-07-08 jrick worktree->root_fd, relpath, ct->blob_id->sha1,
2163 3d1b16d1 2023-07-08 jrick new_base_commit_id->sha1, 1);
2164 3d1b16d1 2023-07-08 jrick if (err) {
2165 3d1b16d1 2023-07-08 jrick got_fileindex_entry_free(ie);
2166 3d1b16d1 2023-07-08 jrick goto done;
2167 3d1b16d1 2023-07-08 jrick }
2168 3d1b16d1 2023-07-08 jrick err = got_fileindex_entry_add(fileindex, ie);
2169 3d1b16d1 2023-07-08 jrick if (err) {
2170 3d1b16d1 2023-07-08 jrick got_fileindex_entry_free(ie);
2171 3d1b16d1 2023-07-08 jrick goto done;
2172 3d1b16d1 2023-07-08 jrick }
2173 3d1b16d1 2023-07-08 jrick }
2174 3d1b16d1 2023-07-08 jrick free(relpath);
2175 3d1b16d1 2023-07-08 jrick relpath = NULL;
2176 3d1b16d1 2023-07-08 jrick }
2177 3d1b16d1 2023-07-08 jrick done:
2178 3d1b16d1 2023-07-08 jrick free(relpath);
2179 3d1b16d1 2023-07-08 jrick return err;
2180 3d1b16d1 2023-07-08 jrick }
2181 3d1b16d1 2023-07-08 jrick
2182 3d1b16d1 2023-07-08 jrick static const struct got_error *
2183 3d1b16d1 2023-07-08 jrick check_out_of_date(const char *in_repo_path, unsigned char status,
2184 3d1b16d1 2023-07-08 jrick unsigned char staged_status, struct got_object_id *base_blob_id,
2185 3d1b16d1 2023-07-08 jrick struct got_object_id *base_commit_id,
2186 3d1b16d1 2023-07-08 jrick struct got_object_id *head_commit_id, struct got_repository *repo,
2187 3d1b16d1 2023-07-08 jrick int ood_errcode)
2188 3d1b16d1 2023-07-08 jrick {
2189 3d1b16d1 2023-07-08 jrick const struct got_error *err = NULL;
2190 3d1b16d1 2023-07-08 jrick struct got_commit_object *commit = NULL;
2191 3d1b16d1 2023-07-08 jrick struct got_object_id *id = NULL;
2192 3d1b16d1 2023-07-08 jrick
2193 3d1b16d1 2023-07-08 jrick if (status != GOT_STATUS_ADD && staged_status != GOT_STATUS_ADD) {
2194 3d1b16d1 2023-07-08 jrick /* Trivial case: base commit == head commit */
2195 3d1b16d1 2023-07-08 jrick if (got_object_id_cmp(base_commit_id, head_commit_id) == 0)
2196 3d1b16d1 2023-07-08 jrick return NULL;
2197 3d1b16d1 2023-07-08 jrick /*
2198 3d1b16d1 2023-07-08 jrick * Ensure file content which local changes were based
2199 3d1b16d1 2023-07-08 jrick * on matches file content in the branch head.
2200 3d1b16d1 2023-07-08 jrick */
2201 3d1b16d1 2023-07-08 jrick err = got_object_open_as_commit(&commit, repo, head_commit_id);
2202 3d1b16d1 2023-07-08 jrick if (err)
2203 3d1b16d1 2023-07-08 jrick goto done;
2204 3d1b16d1 2023-07-08 jrick err = got_object_id_by_path(&id, repo, commit, in_repo_path);
2205 3d1b16d1 2023-07-08 jrick if (err) {
2206 3d1b16d1 2023-07-08 jrick if (err->code == GOT_ERR_NO_TREE_ENTRY)
2207 3d1b16d1 2023-07-08 jrick err = got_error(ood_errcode);
2208 3d1b16d1 2023-07-08 jrick goto done;
2209 3d1b16d1 2023-07-08 jrick } else if (got_object_id_cmp(id, base_blob_id) != 0)
2210 3d1b16d1 2023-07-08 jrick err = got_error(ood_errcode);
2211 3d1b16d1 2023-07-08 jrick } else {
2212 3d1b16d1 2023-07-08 jrick /* Require that added files don't exist in the branch head. */
2213 3d1b16d1 2023-07-08 jrick err = got_object_open_as_commit(&commit, repo, head_commit_id);
2214 3d1b16d1 2023-07-08 jrick if (err)
2215 3d1b16d1 2023-07-08 jrick goto done;
2216 3d1b16d1 2023-07-08 jrick err = got_object_id_by_path(&id, repo, commit, in_repo_path);
2217 3d1b16d1 2023-07-08 jrick if (err && err->code != GOT_ERR_NO_TREE_ENTRY)
2218 3d1b16d1 2023-07-08 jrick goto done;
2219 3d1b16d1 2023-07-08 jrick err = id ? got_error(ood_errcode) : NULL;
2220 3d1b16d1 2023-07-08 jrick }
2221 3d1b16d1 2023-07-08 jrick done:
2222 3d1b16d1 2023-07-08 jrick free(id);
2223 3d1b16d1 2023-07-08 jrick if (commit)
2224 3d1b16d1 2023-07-08 jrick got_object_commit_close(commit);
2225 3d1b16d1 2023-07-08 jrick return err;
2226 3d1b16d1 2023-07-08 jrick }
2227 3d1b16d1 2023-07-08 jrick
2228 3d1b16d1 2023-07-08 jrick static const struct got_error *
2229 3d1b16d1 2023-07-08 jrick commit_worktree(struct got_object_id **new_commit_id,
2230 3d1b16d1 2023-07-08 jrick struct got_pathlist_head *commitable_paths,
2231 3d1b16d1 2023-07-08 jrick struct got_object_id *head_commit_id,
2232 3d1b16d1 2023-07-08 jrick struct got_object_id *parent_id2,
2233 3d1b16d1 2023-07-08 jrick struct got_worktree *worktree,
2234 3d1b16d1 2023-07-08 jrick const char *author, const char *committer, char *diff_path,
2235 3d1b16d1 2023-07-08 jrick got_worktree_commit_msg_cb commit_msg_cb, void *commit_arg,
2236 3d1b16d1 2023-07-08 jrick got_worktree_status_cb status_cb, void *status_arg,
2237 3d1b16d1 2023-07-08 jrick struct got_repository *repo)
2238 3d1b16d1 2023-07-08 jrick {
2239 3d1b16d1 2023-07-08 jrick const struct got_error *err = NULL;
2240 3d1b16d1 2023-07-08 jrick struct got_pathlist_entry *pe;
2241 3d1b16d1 2023-07-08 jrick struct got_commit_object *head_commit = NULL;
2242 3d1b16d1 2023-07-08 jrick struct got_tree_object *head_tree = NULL;
2243 3d1b16d1 2023-07-08 jrick struct got_object_id *new_tree_id = NULL;
2244 3d1b16d1 2023-07-08 jrick int nentries, nparents = 0;
2245 3d1b16d1 2023-07-08 jrick struct got_object_id_queue parent_ids;
2246 3d1b16d1 2023-07-08 jrick struct got_object_qid *pid = NULL;
2247 3d1b16d1 2023-07-08 jrick char *logmsg = NULL;
2248 3d1b16d1 2023-07-08 jrick time_t timestamp;
2249 3d1b16d1 2023-07-08 jrick
2250 3d1b16d1 2023-07-08 jrick *new_commit_id = NULL;
2251 3d1b16d1 2023-07-08 jrick
2252 3d1b16d1 2023-07-08 jrick STAILQ_INIT(&parent_ids);
2253 3d1b16d1 2023-07-08 jrick
2254 3d1b16d1 2023-07-08 jrick err = got_object_open_as_commit(&head_commit, repo, head_commit_id);
2255 3d1b16d1 2023-07-08 jrick if (err)
2256 3d1b16d1 2023-07-08 jrick goto done;
2257 3d1b16d1 2023-07-08 jrick
2258 3d1b16d1 2023-07-08 jrick err = got_object_open_as_tree(&head_tree, repo, head_commit->tree_id);
2259 3d1b16d1 2023-07-08 jrick if (err)
2260 3d1b16d1 2023-07-08 jrick goto done;
2261 3d1b16d1 2023-07-08 jrick
2262 3d1b16d1 2023-07-08 jrick if (commit_msg_cb != NULL) {
2263 3d1b16d1 2023-07-08 jrick err = commit_msg_cb(commitable_paths, diff_path,
2264 3d1b16d1 2023-07-08 jrick &logmsg, commit_arg);
2265 3d1b16d1 2023-07-08 jrick if (err)
2266 3d1b16d1 2023-07-08 jrick goto done;
2267 3d1b16d1 2023-07-08 jrick }
2268 3d1b16d1 2023-07-08 jrick
2269 3d1b16d1 2023-07-08 jrick if (logmsg == NULL || strlen(logmsg) == 0) {
2270 3d1b16d1 2023-07-08 jrick err = got_error(GOT_ERR_COMMIT_MSG_EMPTY);
2271 3d1b16d1 2023-07-08 jrick goto done;
2272 3d1b16d1 2023-07-08 jrick }
2273 3d1b16d1 2023-07-08 jrick
2274 3d1b16d1 2023-07-08 jrick /* Create blobs from added and modified files and record their IDs. */
2275 3d1b16d1 2023-07-08 jrick TAILQ_FOREACH(pe, commitable_paths, entry) {
2276 3d1b16d1 2023-07-08 jrick struct got_commitable *ct = pe->data;
2277 3d1b16d1 2023-07-08 jrick char *ondisk_path;
2278 3d1b16d1 2023-07-08 jrick
2279 3d1b16d1 2023-07-08 jrick /* Blobs for staged files already exist. */
2280 3d1b16d1 2023-07-08 jrick if (ct->staged_status == GOT_STATUS_ADD ||
2281 3d1b16d1 2023-07-08 jrick ct->staged_status == GOT_STATUS_MODIFY)
2282 3d1b16d1 2023-07-08 jrick continue;
2283 3d1b16d1 2023-07-08 jrick
2284 3d1b16d1 2023-07-08 jrick if (ct->status != GOT_STATUS_ADD &&
2285 3d1b16d1 2023-07-08 jrick ct->status != GOT_STATUS_MODIFY &&
2286 3d1b16d1 2023-07-08 jrick ct->status != GOT_STATUS_MODE_CHANGE &&
2287 3d1b16d1 2023-07-08 jrick ct->status != GOT_STATUS_CONFLICT)
2288 3d1b16d1 2023-07-08 jrick continue;
2289 3d1b16d1 2023-07-08 jrick
2290 3d1b16d1 2023-07-08 jrick if (asprintf(&ondisk_path, "%s/%s",
2291 3d1b16d1 2023-07-08 jrick worktree->root_path, pe->path) == -1) {
2292 3d1b16d1 2023-07-08 jrick err = got_error_from_errno("asprintf");
2293 3d1b16d1 2023-07-08 jrick goto done;
2294 3d1b16d1 2023-07-08 jrick }
2295 3d1b16d1 2023-07-08 jrick err = got_object_blob_create(&ct->blob_id, ondisk_path, repo);
2296 3d1b16d1 2023-07-08 jrick free(ondisk_path);
2297 3d1b16d1 2023-07-08 jrick if (err)
2298 3d1b16d1 2023-07-08 jrick goto done;
2299 3d1b16d1 2023-07-08 jrick }
2300 3d1b16d1 2023-07-08 jrick
2301 3d1b16d1 2023-07-08 jrick /* Recursively write new tree objects. */
2302 3d1b16d1 2023-07-08 jrick err = write_tree(&new_tree_id, &nentries, head_tree, "/",
2303 3d1b16d1 2023-07-08 jrick commitable_paths, status_cb, status_arg, repo);
2304 3d1b16d1 2023-07-08 jrick if (err)
2305 3d1b16d1 2023-07-08 jrick goto done;
2306 3d1b16d1 2023-07-08 jrick
2307 3d1b16d1 2023-07-08 jrick err = got_object_qid_alloc(&pid, head_commit_id);
2308 3d1b16d1 2023-07-08 jrick if (err)
2309 3d1b16d1 2023-07-08 jrick goto done;
2310 3d1b16d1 2023-07-08 jrick STAILQ_INSERT_TAIL(&parent_ids, pid, entry);
2311 3d1b16d1 2023-07-08 jrick nparents++;
2312 3d1b16d1 2023-07-08 jrick if (parent_id2) {
2313 3d1b16d1 2023-07-08 jrick err = got_object_qid_alloc(&pid, parent_id2);
2314 3d1b16d1 2023-07-08 jrick if (err)
2315 3d1b16d1 2023-07-08 jrick goto done;
2316 3d1b16d1 2023-07-08 jrick STAILQ_INSERT_TAIL(&parent_ids, pid, entry);
2317 3d1b16d1 2023-07-08 jrick nparents++;
2318 3d1b16d1 2023-07-08 jrick }
2319 3d1b16d1 2023-07-08 jrick timestamp = time(NULL);
2320 3d1b16d1 2023-07-08 jrick err = got_object_commit_create(new_commit_id, new_tree_id, &parent_ids,
2321 3d1b16d1 2023-07-08 jrick nparents, author, timestamp, committer, timestamp, logmsg, repo);
2322 3d1b16d1 2023-07-08 jrick if (logmsg != NULL)
2323 3d1b16d1 2023-07-08 jrick free(logmsg);
2324 3d1b16d1 2023-07-08 jrick if (err)
2325 3d1b16d1 2023-07-08 jrick goto done;
2326 3d1b16d1 2023-07-08 jrick done:
2327 3d1b16d1 2023-07-08 jrick got_object_id_queue_free(&parent_ids);
2328 3d1b16d1 2023-07-08 jrick if (head_tree)
2329 3d1b16d1 2023-07-08 jrick got_object_tree_close(head_tree);
2330 3d1b16d1 2023-07-08 jrick if (head_commit)
2331 3d1b16d1 2023-07-08 jrick got_object_commit_close(head_commit);
2332 3d1b16d1 2023-07-08 jrick return err;
2333 3d1b16d1 2023-07-08 jrick }
2334 3d1b16d1 2023-07-08 jrick
2335 3d1b16d1 2023-07-08 jrick static const struct got_error *
2336 3d1b16d1 2023-07-08 jrick check_path_is_commitable(const char *path,
2337 3d1b16d1 2023-07-08 jrick struct got_pathlist_head *commitable_paths)
2338 3d1b16d1 2023-07-08 jrick {
2339 3d1b16d1 2023-07-08 jrick struct got_pathlist_entry *cpe = NULL;
2340 3d1b16d1 2023-07-08 jrick size_t path_len = strlen(path);
2341 3d1b16d1 2023-07-08 jrick
2342 3d1b16d1 2023-07-08 jrick TAILQ_FOREACH(cpe, commitable_paths, entry) {
2343 3d1b16d1 2023-07-08 jrick struct got_commitable *ct = cpe->data;
2344 3d1b16d1 2023-07-08 jrick const char *ct_path = ct->path;
2345 3d1b16d1 2023-07-08 jrick
2346 3d1b16d1 2023-07-08 jrick while (ct_path[0] == '/')
2347 3d1b16d1 2023-07-08 jrick ct_path++;
2348 3d1b16d1 2023-07-08 jrick
2349 3d1b16d1 2023-07-08 jrick if (strcmp(path, ct_path) == 0 ||
2350 3d1b16d1 2023-07-08 jrick got_path_is_child(ct_path, path, path_len))
2351 3d1b16d1 2023-07-08 jrick break;
2352 3d1b16d1 2023-07-08 jrick }
2353 3d1b16d1 2023-07-08 jrick
2354 3d1b16d1 2023-07-08 jrick if (cpe == NULL)
2355 3d1b16d1 2023-07-08 jrick return got_error_path(path, GOT_ERR_BAD_PATH);
2356 3d1b16d1 2023-07-08 jrick
2357 3d1b16d1 2023-07-08 jrick return NULL;
2358 3d1b16d1 2023-07-08 jrick }
2359 3d1b16d1 2023-07-08 jrick
2360 3d1b16d1 2023-07-08 jrick static const struct got_error *
2361 3d1b16d1 2023-07-08 jrick check_staged_file(void *arg, struct got_fileindex_entry *ie)
2362 3d1b16d1 2023-07-08 jrick {
2363 3d1b16d1 2023-07-08 jrick int *have_staged_files = arg;
2364 3d1b16d1 2023-07-08 jrick
2365 3d1b16d1 2023-07-08 jrick if (got_fileindex_entry_stage_get(ie) != GOT_FILEIDX_STAGE_NONE) {
2366 3d1b16d1 2023-07-08 jrick *have_staged_files = 1;
2367 3d1b16d1 2023-07-08 jrick return got_error(GOT_ERR_CANCELLED);
2368 3d1b16d1 2023-07-08 jrick }
2369 3d1b16d1 2023-07-08 jrick
2370 3d1b16d1 2023-07-08 jrick return NULL;
2371 3d1b16d1 2023-07-08 jrick }
2372 3d1b16d1 2023-07-08 jrick
2373 3d1b16d1 2023-07-08 jrick static const struct got_error *
2374 3d1b16d1 2023-07-08 jrick check_non_staged_files(struct got_fileindex *fileindex,
2375 3d1b16d1 2023-07-08 jrick struct got_pathlist_head *paths)
2376 3d1b16d1 2023-07-08 jrick {
2377 3d1b16d1 2023-07-08 jrick struct got_pathlist_entry *pe;
2378 3d1b16d1 2023-07-08 jrick struct got_fileindex_entry *ie;
2379 3d1b16d1 2023-07-08 jrick
2380 3d1b16d1 2023-07-08 jrick TAILQ_FOREACH(pe, paths, entry) {
2381 3d1b16d1 2023-07-08 jrick if (pe->path[0] == '\0')
2382 3d1b16d1 2023-07-08 jrick continue;
2383 3d1b16d1 2023-07-08 jrick ie = got_fileindex_entry_get(fileindex, pe->path, pe->path_len);
2384 3d1b16d1 2023-07-08 jrick if (ie == NULL)
2385 3d1b16d1 2023-07-08 jrick return got_error_path(pe->path, GOT_ERR_BAD_PATH);
2386 3d1b16d1 2023-07-08 jrick if (got_fileindex_entry_stage_get(ie) == GOT_FILEIDX_STAGE_NONE)
2387 3d1b16d1 2023-07-08 jrick return got_error_path(pe->path,
2388 3d1b16d1 2023-07-08 jrick GOT_ERR_FILE_NOT_STAGED);
2389 3d1b16d1 2023-07-08 jrick }
2390 3d1b16d1 2023-07-08 jrick
2391 3d1b16d1 2023-07-08 jrick return NULL;
2392 3d1b16d1 2023-07-08 jrick }
2393 3d1b16d1 2023-07-08 jrick
2394 3d1b16d1 2023-07-08 jrick static void
2395 3d1b16d1 2023-07-08 jrick print_load_info(int print_colored, int print_found, int print_trees,
2396 3d1b16d1 2023-07-08 jrick int ncolored, int nfound, int ntrees)
2397 3d1b16d1 2023-07-08 jrick {
2398 3d1b16d1 2023-07-08 jrick if (print_colored) {
2399 3d1b16d1 2023-07-08 jrick printf("%d commit%s colored", ncolored,
2400 3d1b16d1 2023-07-08 jrick ncolored == 1 ? "" : "s");
2401 3d1b16d1 2023-07-08 jrick }
2402 3d1b16d1 2023-07-08 jrick if (print_found) {
2403 3d1b16d1 2023-07-08 jrick printf("%s%d object%s found",
2404 3d1b16d1 2023-07-08 jrick ncolored > 0 ? "; " : "",
2405 3d1b16d1 2023-07-08 jrick nfound, nfound == 1 ? "" : "s");
2406 3d1b16d1 2023-07-08 jrick }
2407 3d1b16d1 2023-07-08 jrick if (print_trees) {
2408 3d1b16d1 2023-07-08 jrick printf("; %d tree%s scanned", ntrees,
2409 3d1b16d1 2023-07-08 jrick ntrees == 1 ? "" : "s");
2410 3d1b16d1 2023-07-08 jrick }
2411 3d1b16d1 2023-07-08 jrick }
2412 3d1b16d1 2023-07-08 jrick
2413 3d1b16d1 2023-07-08 jrick struct got_send_progress_arg {
2414 3d1b16d1 2023-07-08 jrick char last_scaled_packsize[FMT_SCALED_STRSIZE];
2415 3d1b16d1 2023-07-08 jrick int verbosity;
2416 3d1b16d1 2023-07-08 jrick int last_ncolored;
2417 3d1b16d1 2023-07-08 jrick int last_nfound;
2418 3d1b16d1 2023-07-08 jrick int last_ntrees;
2419 3d1b16d1 2023-07-08 jrick int loading_done;
2420 3d1b16d1 2023-07-08 jrick int last_ncommits;
2421 3d1b16d1 2023-07-08 jrick int last_nobj_total;
2422 3d1b16d1 2023-07-08 jrick int last_p_deltify;
2423 3d1b16d1 2023-07-08 jrick int last_p_written;
2424 3d1b16d1 2023-07-08 jrick int last_p_sent;
2425 3d1b16d1 2023-07-08 jrick int printed_something;
2426 3d1b16d1 2023-07-08 jrick int sent_something;
2427 3d1b16d1 2023-07-08 jrick struct got_pathlist_head *delete_branches;
2428 3d1b16d1 2023-07-08 jrick };
2429 3d1b16d1 2023-07-08 jrick
2430 3d1b16d1 2023-07-08 jrick static const struct got_error *
2431 3d1b16d1 2023-07-08 jrick send_progress(void *arg, int ncolored, int nfound, int ntrees,
2432 3d1b16d1 2023-07-08 jrick off_t packfile_size, int ncommits, int nobj_total, int nobj_deltify,
2433 3d1b16d1 2023-07-08 jrick int nobj_written, off_t bytes_sent, const char *refname,
2434 3d1b16d1 2023-07-08 jrick const char *errmsg, int success)
2435 3d1b16d1 2023-07-08 jrick {
2436 3d1b16d1 2023-07-08 jrick struct got_send_progress_arg *a = arg;
2437 3d1b16d1 2023-07-08 jrick char scaled_packsize[FMT_SCALED_STRSIZE];
2438 3d1b16d1 2023-07-08 jrick char scaled_sent[FMT_SCALED_STRSIZE];
2439 3d1b16d1 2023-07-08 jrick int p_deltify = 0, p_written = 0, p_sent = 0;
2440 3d1b16d1 2023-07-08 jrick int print_colored = 0, print_found = 0, print_trees = 0;
2441 3d1b16d1 2023-07-08 jrick int print_searching = 0, print_total = 0;
2442 3d1b16d1 2023-07-08 jrick int print_deltify = 0, print_written = 0, print_sent = 0;
2443 3d1b16d1 2023-07-08 jrick
2444 3d1b16d1 2023-07-08 jrick if (a->verbosity < 0)
2445 3d1b16d1 2023-07-08 jrick return NULL;
2446 3d1b16d1 2023-07-08 jrick
2447 3d1b16d1 2023-07-08 jrick if (refname) {
2448 3d1b16d1 2023-07-08 jrick const char *status = success ? "accepted" : "rejected";
2449 3d1b16d1 2023-07-08 jrick
2450 3d1b16d1 2023-07-08 jrick if (success) {
2451 3d1b16d1 2023-07-08 jrick struct got_pathlist_entry *pe;
2452 3d1b16d1 2023-07-08 jrick TAILQ_FOREACH(pe, a->delete_branches, entry) {
2453 3d1b16d1 2023-07-08 jrick const char *branchname = pe->path;
2454 3d1b16d1 2023-07-08 jrick if (got_path_cmp(branchname, refname,
2455 3d1b16d1 2023-07-08 jrick strlen(branchname), strlen(refname)) == 0) {
2456 3d1b16d1 2023-07-08 jrick status = "deleted";
2457 3d1b16d1 2023-07-08 jrick a->sent_something = 1;
2458 3d1b16d1 2023-07-08 jrick break;
2459 3d1b16d1 2023-07-08 jrick }
2460 3d1b16d1 2023-07-08 jrick }
2461 3d1b16d1 2023-07-08 jrick }
2462 3d1b16d1 2023-07-08 jrick
2463 3d1b16d1 2023-07-08 jrick if (a->printed_something)
2464 3d1b16d1 2023-07-08 jrick putchar('\n');
2465 3d1b16d1 2023-07-08 jrick printf("Server has %s %s", status, refname);
2466 3d1b16d1 2023-07-08 jrick if (errmsg)
2467 3d1b16d1 2023-07-08 jrick printf(": %s", errmsg);
2468 3d1b16d1 2023-07-08 jrick a->printed_something = 1;
2469 3d1b16d1 2023-07-08 jrick return NULL;
2470 3d1b16d1 2023-07-08 jrick }
2471 3d1b16d1 2023-07-08 jrick
2472 3d1b16d1 2023-07-08 jrick if (a->last_ncolored != ncolored) {
2473 3d1b16d1 2023-07-08 jrick print_colored = 1;
2474 3d1b16d1 2023-07-08 jrick a->last_ncolored = ncolored;
2475 3d1b16d1 2023-07-08 jrick }
2476 3d1b16d1 2023-07-08 jrick
2477 3d1b16d1 2023-07-08 jrick if (a->last_nfound != nfound) {
2478 3d1b16d1 2023-07-08 jrick print_colored = 1;
2479 3d1b16d1 2023-07-08 jrick print_found = 1;
2480 3d1b16d1 2023-07-08 jrick a->last_nfound = nfound;
2481 3d1b16d1 2023-07-08 jrick }
2482 3d1b16d1 2023-07-08 jrick
2483 3d1b16d1 2023-07-08 jrick if (a->last_ntrees != ntrees) {
2484 3d1b16d1 2023-07-08 jrick print_colored = 1;
2485 3d1b16d1 2023-07-08 jrick print_found = 1;
2486 3d1b16d1 2023-07-08 jrick print_trees = 1;
2487 3d1b16d1 2023-07-08 jrick a->last_ntrees = ntrees;
2488 3d1b16d1 2023-07-08 jrick }
2489 3d1b16d1 2023-07-08 jrick
2490 3d1b16d1 2023-07-08 jrick if ((print_colored || print_found || print_trees) &&
2491 3d1b16d1 2023-07-08 jrick !a->loading_done) {
2492 3d1b16d1 2023-07-08 jrick printf("\r");
2493 3d1b16d1 2023-07-08 jrick print_load_info(print_colored, print_found, print_trees,
2494 3d1b16d1 2023-07-08 jrick ncolored, nfound, ntrees);
2495 3d1b16d1 2023-07-08 jrick a->printed_something = 1;
2496 3d1b16d1 2023-07-08 jrick fflush(stdout);
2497 3d1b16d1 2023-07-08 jrick return NULL;
2498 3d1b16d1 2023-07-08 jrick } else if (!a->loading_done) {
2499 3d1b16d1 2023-07-08 jrick printf("\r");
2500 3d1b16d1 2023-07-08 jrick print_load_info(1, 1, 1, ncolored, nfound, ntrees);
2501 3d1b16d1 2023-07-08 jrick printf("\n");
2502 3d1b16d1 2023-07-08 jrick a->loading_done = 1;
2503 3d1b16d1 2023-07-08 jrick }
2504 3d1b16d1 2023-07-08 jrick
2505 3d1b16d1 2023-07-08 jrick if (fmt_scaled(packfile_size, scaled_packsize) == -1)
2506 3d1b16d1 2023-07-08 jrick return got_error_from_errno("fmt_scaled");
2507 3d1b16d1 2023-07-08 jrick if (fmt_scaled(bytes_sent, scaled_sent) == -1)
2508 3d1b16d1 2023-07-08 jrick return got_error_from_errno("fmt_scaled");
2509 3d1b16d1 2023-07-08 jrick
2510 3d1b16d1 2023-07-08 jrick if (a->last_ncommits != ncommits) {
2511 3d1b16d1 2023-07-08 jrick print_searching = 1;
2512 3d1b16d1 2023-07-08 jrick a->last_ncommits = ncommits;
2513 3d1b16d1 2023-07-08 jrick }
2514 3d1b16d1 2023-07-08 jrick
2515 3d1b16d1 2023-07-08 jrick if (a->last_nobj_total != nobj_total) {
2516 3d1b16d1 2023-07-08 jrick print_searching = 1;
2517 3d1b16d1 2023-07-08 jrick print_total = 1;
2518 3d1b16d1 2023-07-08 jrick a->last_nobj_total = nobj_total;
2519 3d1b16d1 2023-07-08 jrick }
2520 3d1b16d1 2023-07-08 jrick
2521 3d1b16d1 2023-07-08 jrick if (packfile_size > 0 && (a->last_scaled_packsize[0] == '\0' ||
2522 3d1b16d1 2023-07-08 jrick strcmp(scaled_packsize, a->last_scaled_packsize)) != 0) {
2523 3d1b16d1 2023-07-08 jrick if (strlcpy(a->last_scaled_packsize, scaled_packsize,
2524 3d1b16d1 2023-07-08 jrick FMT_SCALED_STRSIZE) >= FMT_SCALED_STRSIZE)
2525 3d1b16d1 2023-07-08 jrick return got_error(GOT_ERR_NO_SPACE);
2526 3d1b16d1 2023-07-08 jrick }
2527 3d1b16d1 2023-07-08 jrick
2528 3d1b16d1 2023-07-08 jrick if (nobj_deltify > 0 || nobj_written > 0) {
2529 3d1b16d1 2023-07-08 jrick if (nobj_deltify > 0) {
2530 3d1b16d1 2023-07-08 jrick p_deltify = (nobj_deltify * 100) / nobj_total;
2531 3d1b16d1 2023-07-08 jrick if (p_deltify != a->last_p_deltify) {
2532 3d1b16d1 2023-07-08 jrick a->last_p_deltify = p_deltify;
2533 3d1b16d1 2023-07-08 jrick print_searching = 1;
2534 3d1b16d1 2023-07-08 jrick print_total = 1;
2535 3d1b16d1 2023-07-08 jrick print_deltify = 1;
2536 3d1b16d1 2023-07-08 jrick }
2537 3d1b16d1 2023-07-08 jrick }
2538 3d1b16d1 2023-07-08 jrick if (nobj_written > 0) {
2539 3d1b16d1 2023-07-08 jrick p_written = (nobj_written * 100) / nobj_total;
2540 3d1b16d1 2023-07-08 jrick if (p_written != a->last_p_written) {
2541 3d1b16d1 2023-07-08 jrick a->last_p_written = p_written;
2542 3d1b16d1 2023-07-08 jrick print_searching = 1;
2543 3d1b16d1 2023-07-08 jrick print_total = 1;
2544 3d1b16d1 2023-07-08 jrick print_deltify = 1;
2545 3d1b16d1 2023-07-08 jrick print_written = 1;
2546 3d1b16d1 2023-07-08 jrick }
2547 3d1b16d1 2023-07-08 jrick }
2548 3d1b16d1 2023-07-08 jrick }
2549 3d1b16d1 2023-07-08 jrick
2550 3d1b16d1 2023-07-08 jrick if (bytes_sent > 0) {
2551 3d1b16d1 2023-07-08 jrick p_sent = (bytes_sent * 100) / packfile_size;
2552 3d1b16d1 2023-07-08 jrick if (p_sent != a->last_p_sent) {
2553 3d1b16d1 2023-07-08 jrick a->last_p_sent = p_sent;
2554 3d1b16d1 2023-07-08 jrick print_searching = 1;
2555 3d1b16d1 2023-07-08 jrick print_total = 1;
2556 3d1b16d1 2023-07-08 jrick print_deltify = 1;
2557 3d1b16d1 2023-07-08 jrick print_written = 1;
2558 3d1b16d1 2023-07-08 jrick print_sent = 1;
2559 3d1b16d1 2023-07-08 jrick }
2560 3d1b16d1 2023-07-08 jrick a->sent_something = 1;
2561 3d1b16d1 2023-07-08 jrick }
2562 3d1b16d1 2023-07-08 jrick
2563 3d1b16d1 2023-07-08 jrick if (print_searching || print_total || print_deltify || print_written ||
2564 3d1b16d1 2023-07-08 jrick print_sent)
2565 3d1b16d1 2023-07-08 jrick printf("\r");
2566 3d1b16d1 2023-07-08 jrick if (print_searching)
2567 3d1b16d1 2023-07-08 jrick printf("packing %d reference%s", ncommits,
2568 3d1b16d1 2023-07-08 jrick ncommits == 1 ? "" : "s");
2569 3d1b16d1 2023-07-08 jrick if (print_total)
2570 3d1b16d1 2023-07-08 jrick printf("; %d object%s", nobj_total,
2571 3d1b16d1 2023-07-08 jrick nobj_total == 1 ? "" : "s");
2572 3d1b16d1 2023-07-08 jrick if (print_deltify)
2573 3d1b16d1 2023-07-08 jrick printf("; deltify: %d%%", p_deltify);
2574 3d1b16d1 2023-07-08 jrick if (print_sent)
2575 3d1b16d1 2023-07-08 jrick printf("; uploading pack: %*s %d%%", FMT_SCALED_STRSIZE - 2,
2576 3d1b16d1 2023-07-08 jrick scaled_packsize, p_sent);
2577 3d1b16d1 2023-07-08 jrick else if (print_written)
2578 3d1b16d1 2023-07-08 jrick printf("; writing pack: %*s %d%%", FMT_SCALED_STRSIZE - 2,
2579 3d1b16d1 2023-07-08 jrick scaled_packsize, p_written);
2580 3d1b16d1 2023-07-08 jrick if (print_searching || print_total || print_deltify ||
2581 3d1b16d1 2023-07-08 jrick print_written || print_sent) {
2582 3d1b16d1 2023-07-08 jrick a->printed_something = 1;
2583 3d1b16d1 2023-07-08 jrick fflush(stdout);
2584 3d1b16d1 2023-07-08 jrick }
2585 3d1b16d1 2023-07-08 jrick return NULL;
2586 3d1b16d1 2023-07-08 jrick }
2587 3d1b16d1 2023-07-08 jrick
2588 3d1b16d1 2023-07-08 jrick struct got_fetch_progress_arg {
2589 3d1b16d1 2023-07-08 jrick char last_scaled_size[FMT_SCALED_STRSIZE];
2590 3d1b16d1 2023-07-08 jrick int last_p_indexed;
2591 3d1b16d1 2023-07-08 jrick int last_p_resolved;
2592 3d1b16d1 2023-07-08 jrick int verbosity;
2593 3d1b16d1 2023-07-08 jrick
2594 3d1b16d1 2023-07-08 jrick struct got_repository *repo;
2595 3d1b16d1 2023-07-08 jrick };
2596 3d1b16d1 2023-07-08 jrick
2597 3d1b16d1 2023-07-08 jrick static const struct got_error *
2598 3d1b16d1 2023-07-08 jrick fetch_progress(void *arg, const char *message, off_t packfile_size,
2599 3d1b16d1 2023-07-08 jrick int nobj_total, int nobj_indexed, int nobj_loose, int nobj_resolved)
2600 3d1b16d1 2023-07-08 jrick {
2601 3d1b16d1 2023-07-08 jrick struct got_fetch_progress_arg *a = arg;
2602 3d1b16d1 2023-07-08 jrick char scaled_size[FMT_SCALED_STRSIZE];
2603 3d1b16d1 2023-07-08 jrick int p_indexed, p_resolved;
2604 3d1b16d1 2023-07-08 jrick int print_size = 0, print_indexed = 0, print_resolved = 0;
2605 3d1b16d1 2023-07-08 jrick
2606 3d1b16d1 2023-07-08 jrick if (a->verbosity < 0)
2607 3d1b16d1 2023-07-08 jrick return NULL;
2608 3d1b16d1 2023-07-08 jrick
2609 3d1b16d1 2023-07-08 jrick if (message && message[0] != '\0') {
2610 3d1b16d1 2023-07-08 jrick printf("\rserver: %s", message);
2611 3d1b16d1 2023-07-08 jrick fflush(stdout);
2612 3d1b16d1 2023-07-08 jrick return NULL;
2613 3d1b16d1 2023-07-08 jrick }
2614 3d1b16d1 2023-07-08 jrick
2615 3d1b16d1 2023-07-08 jrick if (packfile_size > 0 || nobj_indexed > 0) {
2616 3d1b16d1 2023-07-08 jrick if (fmt_scaled(packfile_size, scaled_size) == 0 &&
2617 3d1b16d1 2023-07-08 jrick (a->last_scaled_size[0] == '\0' ||
2618 3d1b16d1 2023-07-08 jrick strcmp(scaled_size, a->last_scaled_size)) != 0) {
2619 3d1b16d1 2023-07-08 jrick print_size = 1;
2620 3d1b16d1 2023-07-08 jrick if (strlcpy(a->last_scaled_size, scaled_size,
2621 3d1b16d1 2023-07-08 jrick FMT_SCALED_STRSIZE) >= FMT_SCALED_STRSIZE)
2622 3d1b16d1 2023-07-08 jrick return got_error(GOT_ERR_NO_SPACE);
2623 3d1b16d1 2023-07-08 jrick }
2624 3d1b16d1 2023-07-08 jrick if (nobj_indexed > 0) {
2625 3d1b16d1 2023-07-08 jrick p_indexed = (nobj_indexed * 100) / nobj_total;
2626 3d1b16d1 2023-07-08 jrick if (p_indexed != a->last_p_indexed) {
2627 3d1b16d1 2023-07-08 jrick a->last_p_indexed = p_indexed;
2628 3d1b16d1 2023-07-08 jrick print_indexed = 1;
2629 3d1b16d1 2023-07-08 jrick print_size = 1;
2630 3d1b16d1 2023-07-08 jrick }
2631 3d1b16d1 2023-07-08 jrick }
2632 3d1b16d1 2023-07-08 jrick if (nobj_resolved > 0) {
2633 3d1b16d1 2023-07-08 jrick p_resolved = (nobj_resolved * 100) /
2634 3d1b16d1 2023-07-08 jrick (nobj_total - nobj_loose);
2635 3d1b16d1 2023-07-08 jrick if (p_resolved != a->last_p_resolved) {
2636 3d1b16d1 2023-07-08 jrick a->last_p_resolved = p_resolved;
2637 3d1b16d1 2023-07-08 jrick print_resolved = 1;
2638 3d1b16d1 2023-07-08 jrick print_indexed = 1;
2639 3d1b16d1 2023-07-08 jrick print_size = 1;
2640 3d1b16d1 2023-07-08 jrick }
2641 3d1b16d1 2023-07-08 jrick }
2642 3d1b16d1 2023-07-08 jrick
2643 3d1b16d1 2023-07-08 jrick }
2644 3d1b16d1 2023-07-08 jrick if (print_size || print_indexed || print_resolved)
2645 3d1b16d1 2023-07-08 jrick printf("\r");
2646 3d1b16d1 2023-07-08 jrick if (print_size)
2647 3d1b16d1 2023-07-08 jrick printf("%*s fetched", FMT_SCALED_STRSIZE - 2, scaled_size);
2648 3d1b16d1 2023-07-08 jrick if (print_indexed)
2649 3d1b16d1 2023-07-08 jrick printf("; indexing %d%%", p_indexed);
2650 3d1b16d1 2023-07-08 jrick if (print_resolved)
2651 3d1b16d1 2023-07-08 jrick printf("; resolving deltas %d%%", p_resolved);
2652 3d1b16d1 2023-07-08 jrick if (print_size || print_indexed || print_resolved) {
2653 3d1b16d1 2023-07-08 jrick putchar('\n');
2654 3d1b16d1 2023-07-08 jrick fflush(stdout);
2655 3d1b16d1 2023-07-08 jrick }
2656 3d1b16d1 2023-07-08 jrick
2657 3d1b16d1 2023-07-08 jrick return NULL;
2658 3d1b16d1 2023-07-08 jrick }
2659 3d1b16d1 2023-07-08 jrick
2660 3d1b16d1 2023-07-08 jrick static const struct got_error *
2661 3d1b16d1 2023-07-08 jrick create_symref(const char *refname, struct got_reference *target_ref,
2662 3d1b16d1 2023-07-08 jrick int verbosity, struct got_repository *repo)
2663 3d1b16d1 2023-07-08 jrick {
2664 3d1b16d1 2023-07-08 jrick const struct got_error *err;
2665 3d1b16d1 2023-07-08 jrick struct got_reference *head_symref;
2666 3d1b16d1 2023-07-08 jrick
2667 3d1b16d1 2023-07-08 jrick err = got_ref_alloc_symref(&head_symref, refname, target_ref);
2668 3d1b16d1 2023-07-08 jrick if (err)
2669 3d1b16d1 2023-07-08 jrick return err;
2670 3d1b16d1 2023-07-08 jrick
2671 3d1b16d1 2023-07-08 jrick err = got_ref_write(head_symref, repo);
2672 3d1b16d1 2023-07-08 jrick if (err == NULL && verbosity > 0) {
2673 3d1b16d1 2023-07-08 jrick printf("Created reference %s: %s\n", GOT_REF_HEAD,
2674 3d1b16d1 2023-07-08 jrick got_ref_get_name(target_ref));
2675 3d1b16d1 2023-07-08 jrick }
2676 3d1b16d1 2023-07-08 jrick got_ref_close(head_symref);
2677 3d1b16d1 2023-07-08 jrick return err;
2678 3d1b16d1 2023-07-08 jrick }
2679 3d1b16d1 2023-07-08 jrick
2680 3d1b16d1 2023-07-08 jrick static const struct got_error *
2681 3d1b16d1 2023-07-08 jrick create_ref(const char *refname, struct got_object_id *id,
2682 3d1b16d1 2023-07-08 jrick int verbosity, struct got_repository *repo)
2683 3d1b16d1 2023-07-08 jrick {
2684 3d1b16d1 2023-07-08 jrick const struct got_error *err = NULL;
2685 3d1b16d1 2023-07-08 jrick struct got_reference *ref;
2686 3d1b16d1 2023-07-08 jrick char *id_str;
2687 3d1b16d1 2023-07-08 jrick
2688 3d1b16d1 2023-07-08 jrick err = got_object_id_str(&id_str, id);
2689 3d1b16d1 2023-07-08 jrick if (err)
2690 3d1b16d1 2023-07-08 jrick return err;
2691 3d1b16d1 2023-07-08 jrick
2692 3d1b16d1 2023-07-08 jrick err = got_ref_alloc(&ref, refname, id);
2693 3d1b16d1 2023-07-08 jrick if (err)
2694 3d1b16d1 2023-07-08 jrick goto done;
2695 3d1b16d1 2023-07-08 jrick
2696 3d1b16d1 2023-07-08 jrick err = got_ref_write(ref, repo);
2697 3d1b16d1 2023-07-08 jrick got_ref_close(ref);
2698 3d1b16d1 2023-07-08 jrick
2699 3d1b16d1 2023-07-08 jrick if (err == NULL && verbosity >= 0)
2700 3d1b16d1 2023-07-08 jrick printf("Created reference %s: %s\n", refname, id_str);
2701 3d1b16d1 2023-07-08 jrick done:
2702 3d1b16d1 2023-07-08 jrick free(id_str);
2703 3d1b16d1 2023-07-08 jrick return err;
2704 3d1b16d1 2023-07-08 jrick }
2705 3d1b16d1 2023-07-08 jrick
2706 3d1b16d1 2023-07-08 jrick static const struct got_error *
2707 3d1b16d1 2023-07-08 jrick update_ref(struct got_reference *ref, struct got_object_id *new_id,
2708 3d1b16d1 2023-07-08 jrick int verbosity, struct got_repository *repo)
2709 3d1b16d1 2023-07-08 jrick {
2710 3d1b16d1 2023-07-08 jrick const struct got_error *err = NULL;
2711 3d1b16d1 2023-07-08 jrick char *new_id_str = NULL;
2712 3d1b16d1 2023-07-08 jrick struct got_object_id *old_id = NULL;
2713 3d1b16d1 2023-07-08 jrick
2714 3d1b16d1 2023-07-08 jrick err = got_object_id_str(&new_id_str, new_id);
2715 3d1b16d1 2023-07-08 jrick if (err)
2716 3d1b16d1 2023-07-08 jrick goto done;
2717 3d1b16d1 2023-07-08 jrick
2718 3d1b16d1 2023-07-08 jrick if (strncmp(got_ref_get_name(ref), "refs/tags/", 10) == 0) {
2719 3d1b16d1 2023-07-08 jrick err = got_ref_resolve(&old_id, repo, ref);
2720 3d1b16d1 2023-07-08 jrick if (err)
2721 3d1b16d1 2023-07-08 jrick goto done;
2722 3d1b16d1 2023-07-08 jrick if (got_object_id_cmp(old_id, new_id) == 0)
2723 3d1b16d1 2023-07-08 jrick goto done;
2724 3d1b16d1 2023-07-08 jrick if (verbosity >= 0) {
2725 3d1b16d1 2023-07-08 jrick printf("Rejecting update of existing tag %s: %s\n",
2726 3d1b16d1 2023-07-08 jrick got_ref_get_name(ref), new_id_str);
2727 3d1b16d1 2023-07-08 jrick }
2728 3d1b16d1 2023-07-08 jrick goto done;
2729 3d1b16d1 2023-07-08 jrick }
2730 3d1b16d1 2023-07-08 jrick
2731 3d1b16d1 2023-07-08 jrick if (got_ref_is_symbolic(ref)) {
2732 3d1b16d1 2023-07-08 jrick if (verbosity >= 0) {
2733 3d1b16d1 2023-07-08 jrick printf("Replacing reference %s: %s\n",
2734 3d1b16d1 2023-07-08 jrick got_ref_get_name(ref),
2735 3d1b16d1 2023-07-08 jrick got_ref_get_symref_target(ref));
2736 3d1b16d1 2023-07-08 jrick }
2737 3d1b16d1 2023-07-08 jrick err = got_ref_change_symref_to_ref(ref, new_id);
2738 3d1b16d1 2023-07-08 jrick if (err)
2739 3d1b16d1 2023-07-08 jrick goto done;
2740 3d1b16d1 2023-07-08 jrick err = got_ref_write(ref, repo);
2741 3d1b16d1 2023-07-08 jrick if (err)
2742 3d1b16d1 2023-07-08 jrick goto done;
2743 3d1b16d1 2023-07-08 jrick } else {
2744 3d1b16d1 2023-07-08 jrick err = got_ref_resolve(&old_id, repo, ref);
2745 3d1b16d1 2023-07-08 jrick if (err)
2746 3d1b16d1 2023-07-08 jrick goto done;
2747 3d1b16d1 2023-07-08 jrick if (got_object_id_cmp(old_id, new_id) == 0)
2748 3d1b16d1 2023-07-08 jrick goto done;
2749 3d1b16d1 2023-07-08 jrick
2750 3d1b16d1 2023-07-08 jrick err = got_ref_change_ref(ref, new_id);
2751 3d1b16d1 2023-07-08 jrick if (err)
2752 3d1b16d1 2023-07-08 jrick goto done;
2753 3d1b16d1 2023-07-08 jrick err = got_ref_write(ref, repo);
2754 3d1b16d1 2023-07-08 jrick if (err)
2755 3d1b16d1 2023-07-08 jrick goto done;
2756 3d1b16d1 2023-07-08 jrick }
2757 3d1b16d1 2023-07-08 jrick
2758 3d1b16d1 2023-07-08 jrick if (verbosity >= 0)
2759 3d1b16d1 2023-07-08 jrick printf("Updated %s: %s\n", got_ref_get_name(ref),
2760 3d1b16d1 2023-07-08 jrick new_id_str);
2761 3d1b16d1 2023-07-08 jrick done:
2762 3d1b16d1 2023-07-08 jrick free(old_id);
2763 3d1b16d1 2023-07-08 jrick free(new_id_str);
2764 3d1b16d1 2023-07-08 jrick return err;
2765 3d1b16d1 2023-07-08 jrick }
2766 3d1b16d1 2023-07-08 jrick
2767 3d1b16d1 2023-07-08 jrick static const struct got_error *
2768 3d1b16d1 2023-07-08 jrick fetch_updated_remote(const char *proto, const char *host, const char *port,
2769 3d1b16d1 2023-07-08 jrick const char *server_path, int verbosity,
2770 3d1b16d1 2023-07-08 jrick const struct got_remote_repo *remote, struct got_repository *repo,
2771 3d1b16d1 2023-07-08 jrick struct got_reference *head_ref, const char *head_refname)
2772 3d1b16d1 2023-07-08 jrick {
2773 3d1b16d1 2023-07-08 jrick const struct got_error *err = NULL, *unlock_err = NULL;
2774 3d1b16d1 2023-07-08 jrick struct got_pathlist_entry *pe;
2775 3d1b16d1 2023-07-08 jrick struct got_pathlist_head learned_refs;
2776 3d1b16d1 2023-07-08 jrick struct got_pathlist_head symrefs;
2777 3d1b16d1 2023-07-08 jrick struct got_pathlist_head wanted_branches;
2778 3d1b16d1 2023-07-08 jrick struct got_pathlist_head wanted_refs;
2779 3d1b16d1 2023-07-08 jrick struct got_object_id *pack_hash;
2780 3d1b16d1 2023-07-08 jrick struct got_fetch_progress_arg fpa;
2781 3d1b16d1 2023-07-08 jrick int fetchfd = -1;
2782 3d1b16d1 2023-07-08 jrick pid_t fetchpid = -1;
2783 3d1b16d1 2023-07-08 jrick
2784 3d1b16d1 2023-07-08 jrick TAILQ_INIT(&learned_refs);
2785 3d1b16d1 2023-07-08 jrick TAILQ_INIT(&symrefs);
2786 3d1b16d1 2023-07-08 jrick TAILQ_INIT(&wanted_branches);
2787 3d1b16d1 2023-07-08 jrick TAILQ_INIT(&wanted_refs);
2788 3d1b16d1 2023-07-08 jrick
2789 3d1b16d1 2023-07-08 jrick err = got_pathlist_insert(NULL, &wanted_branches, head_refname,
2790 3d1b16d1 2023-07-08 jrick NULL);
2791 3d1b16d1 2023-07-08 jrick if (err)
2792 3d1b16d1 2023-07-08 jrick goto done;
2793 3d1b16d1 2023-07-08 jrick
2794 3d1b16d1 2023-07-08 jrick err = got_fetch_connect(&fetchpid, &fetchfd, proto, host,
2795 3d1b16d1 2023-07-08 jrick port, server_path, verbosity);
2796 3d1b16d1 2023-07-08 jrick if (err)
2797 3d1b16d1 2023-07-08 jrick goto done;
2798 3d1b16d1 2023-07-08 jrick
2799 3d1b16d1 2023-07-08 jrick fpa.last_scaled_size[0] = '\0';
2800 3d1b16d1 2023-07-08 jrick fpa.last_p_indexed = -1;
2801 3d1b16d1 2023-07-08 jrick fpa.last_p_resolved = -1;
2802 3d1b16d1 2023-07-08 jrick fpa.verbosity = verbosity;
2803 3d1b16d1 2023-07-08 jrick fpa.repo = repo;
2804 3d1b16d1 2023-07-08 jrick
2805 3d1b16d1 2023-07-08 jrick err = got_fetch_pack(&pack_hash, &learned_refs, &symrefs,
2806 3d1b16d1 2023-07-08 jrick remote->name, 1, 0, &wanted_branches, &wanted_refs, 0, verbosity,
2807 3d1b16d1 2023-07-08 jrick fetchfd, repo, head_refname, NULL, 0, fetch_progress, &fpa);
2808 3d1b16d1 2023-07-08 jrick if (err)
2809 3d1b16d1 2023-07-08 jrick goto done;
2810 3d1b16d1 2023-07-08 jrick
2811 3d1b16d1 2023-07-08 jrick /* Update references provided with the pack file. */
2812 3d1b16d1 2023-07-08 jrick TAILQ_FOREACH(pe, &learned_refs, entry) {
2813 3d1b16d1 2023-07-08 jrick const char *refname = pe->path;
2814 3d1b16d1 2023-07-08 jrick struct got_object_id *id = pe->data;
2815 3d1b16d1 2023-07-08 jrick struct got_reference *ref;
2816 3d1b16d1 2023-07-08 jrick
2817 3d1b16d1 2023-07-08 jrick err = got_ref_open(&ref, repo, refname, 0);
2818 3d1b16d1 2023-07-08 jrick if (err) {
2819 3d1b16d1 2023-07-08 jrick if (err->code != GOT_ERR_NOT_REF)
2820 3d1b16d1 2023-07-08 jrick goto done;
2821 3d1b16d1 2023-07-08 jrick err = create_ref(refname, id, verbosity, repo);
2822 3d1b16d1 2023-07-08 jrick if (err)
2823 3d1b16d1 2023-07-08 jrick goto done;
2824 3d1b16d1 2023-07-08 jrick } else {
2825 3d1b16d1 2023-07-08 jrick err = update_ref(ref, id, verbosity, repo);
2826 3d1b16d1 2023-07-08 jrick unlock_err = got_ref_unlock(ref);
2827 3d1b16d1 2023-07-08 jrick if (unlock_err && err == NULL)
2828 3d1b16d1 2023-07-08 jrick err = unlock_err;
2829 3d1b16d1 2023-07-08 jrick got_ref_close(ref);
2830 3d1b16d1 2023-07-08 jrick if (err)
2831 3d1b16d1 2023-07-08 jrick goto done;
2832 3d1b16d1 2023-07-08 jrick }
2833 3d1b16d1 2023-07-08 jrick }
2834 3d1b16d1 2023-07-08 jrick
2835 3d1b16d1 2023-07-08 jrick /* Set the HEAD reference if the server provided one. */
2836 3d1b16d1 2023-07-08 jrick TAILQ_FOREACH(pe, &symrefs, entry) {
2837 3d1b16d1 2023-07-08 jrick struct got_reference *target_ref;
2838 3d1b16d1 2023-07-08 jrick const char *refname = pe->path;
2839 3d1b16d1 2023-07-08 jrick const char *target = pe->data;
2840 3d1b16d1 2023-07-08 jrick char *remote_refname = NULL, *remote_target = NULL;
2841 3d1b16d1 2023-07-08 jrick
2842 3d1b16d1 2023-07-08 jrick if (strcmp(refname, GOT_REF_HEAD) != 0)
2843 3d1b16d1 2023-07-08 jrick continue;
2844 3d1b16d1 2023-07-08 jrick
2845 3d1b16d1 2023-07-08 jrick err = got_ref_open(&target_ref, repo, target, 0);
2846 3d1b16d1 2023-07-08 jrick if (err) {
2847 3d1b16d1 2023-07-08 jrick if (err->code == GOT_ERR_NOT_REF) {
2848 3d1b16d1 2023-07-08 jrick err = NULL;
2849 3d1b16d1 2023-07-08 jrick continue;
2850 3d1b16d1 2023-07-08 jrick }
2851 3d1b16d1 2023-07-08 jrick goto done;
2852 3d1b16d1 2023-07-08 jrick }
2853 3d1b16d1 2023-07-08 jrick
2854 3d1b16d1 2023-07-08 jrick err = create_symref(refname, target_ref, verbosity, repo);
2855 3d1b16d1 2023-07-08 jrick got_ref_close(target_ref);
2856 3d1b16d1 2023-07-08 jrick if (err)
2857 3d1b16d1 2023-07-08 jrick goto done;
2858 3d1b16d1 2023-07-08 jrick
2859 3d1b16d1 2023-07-08 jrick if (remote->mirror_references)
2860 3d1b16d1 2023-07-08 jrick continue;
2861 3d1b16d1 2023-07-08 jrick
2862 3d1b16d1 2023-07-08 jrick if (strncmp("refs/heads/", target, 11) != 0)
2863 3d1b16d1 2023-07-08 jrick continue;
2864 3d1b16d1 2023-07-08 jrick
2865 3d1b16d1 2023-07-08 jrick if (asprintf(&remote_refname,
2866 3d1b16d1 2023-07-08 jrick "refs/remotes/%s/%s", GOT_FETCH_DEFAULT_REMOTE_NAME,
2867 3d1b16d1 2023-07-08 jrick refname) == -1) {
2868 3d1b16d1 2023-07-08 jrick err = got_error_from_errno("asprintf");
2869 3d1b16d1 2023-07-08 jrick goto done;
2870 3d1b16d1 2023-07-08 jrick }
2871 3d1b16d1 2023-07-08 jrick if (asprintf(&remote_target,
2872 3d1b16d1 2023-07-08 jrick "refs/remotes/%s/%s", GOT_FETCH_DEFAULT_REMOTE_NAME,
2873 3d1b16d1 2023-07-08 jrick target + 11) == -1) {
2874 3d1b16d1 2023-07-08 jrick err = got_error_from_errno("asprintf");
2875 3d1b16d1 2023-07-08 jrick free(remote_refname);
2876 3d1b16d1 2023-07-08 jrick goto done;
2877 3d1b16d1 2023-07-08 jrick }
2878 3d1b16d1 2023-07-08 jrick err = got_ref_open(&target_ref, repo, remote_target, 0);
2879 3d1b16d1 2023-07-08 jrick if (err) {
2880 3d1b16d1 2023-07-08 jrick free(remote_refname);
2881 3d1b16d1 2023-07-08 jrick free(remote_target);
2882 3d1b16d1 2023-07-08 jrick if (err->code == GOT_ERR_NOT_REF) {
2883 3d1b16d1 2023-07-08 jrick err = NULL;
2884 3d1b16d1 2023-07-08 jrick continue;
2885 3d1b16d1 2023-07-08 jrick }
2886 3d1b16d1 2023-07-08 jrick goto done;
2887 3d1b16d1 2023-07-08 jrick }
2888 3d1b16d1 2023-07-08 jrick err = create_symref(remote_refname, target_ref,
2889 3d1b16d1 2023-07-08 jrick verbosity - 1, repo);
2890 3d1b16d1 2023-07-08 jrick free(remote_refname);
2891 3d1b16d1 2023-07-08 jrick free(remote_target);
2892 3d1b16d1 2023-07-08 jrick got_ref_close(target_ref);
2893 3d1b16d1 2023-07-08 jrick if (err)
2894 3d1b16d1 2023-07-08 jrick goto done;
2895 3d1b16d1 2023-07-08 jrick }
2896 3d1b16d1 2023-07-08 jrick
2897 3d1b16d1 2023-07-08 jrick done:
2898 3d1b16d1 2023-07-08 jrick got_pathlist_free(&learned_refs, GOT_PATHLIST_FREE_NONE);
2899 3d1b16d1 2023-07-08 jrick got_pathlist_free(&symrefs, GOT_PATHLIST_FREE_NONE);
2900 3d1b16d1 2023-07-08 jrick got_pathlist_free(&wanted_branches, GOT_PATHLIST_FREE_NONE);
2901 3d1b16d1 2023-07-08 jrick got_pathlist_free(&wanted_refs, GOT_PATHLIST_FREE_NONE);
2902 3d1b16d1 2023-07-08 jrick return err;
2903 3d1b16d1 2023-07-08 jrick }
2904 3d1b16d1 2023-07-08 jrick
2905 3d1b16d1 2023-07-08 jrick
2906 3d1b16d1 2023-07-08 jrick const struct got_error *
2907 3d1b16d1 2023-07-08 jrick got_worktree_cvg_commit(struct got_object_id **new_commit_id,
2908 3d1b16d1 2023-07-08 jrick struct got_worktree *worktree, struct got_pathlist_head *paths,
2909 3d1b16d1 2023-07-08 jrick const char *author, const char *committer, int allow_bad_symlinks,
2910 3d1b16d1 2023-07-08 jrick int show_diff, int commit_conflicts,
2911 3d1b16d1 2023-07-08 jrick got_worktree_commit_msg_cb commit_msg_cb, void *commit_arg,
2912 3d1b16d1 2023-07-08 jrick got_worktree_status_cb status_cb, void *status_arg,
2913 3d1b16d1 2023-07-08 jrick const char *proto, const char *host, const char *port,
2914 3d1b16d1 2023-07-08 jrick const char *server_path, int verbosity,
2915 3d1b16d1 2023-07-08 jrick const struct got_remote_repo *remote,
2916 3d1b16d1 2023-07-08 jrick got_cancel_cb check_cancelled,
2917 3d1b16d1 2023-07-08 jrick struct got_repository *repo)
2918 3d1b16d1 2023-07-08 jrick {
2919 3d1b16d1 2023-07-08 jrick const struct got_error *err = NULL, *unlockerr = NULL, *sync_err;
2920 3d1b16d1 2023-07-08 jrick struct got_fileindex *fileindex = NULL;
2921 3d1b16d1 2023-07-08 jrick char *fileindex_path = NULL;
2922 3d1b16d1 2023-07-08 jrick struct got_pathlist_head commitable_paths;
2923 3d1b16d1 2023-07-08 jrick struct collect_commitables_arg cc_arg;
2924 3d1b16d1 2023-07-08 jrick struct got_pathlist_entry *pe;
2925 3d1b16d1 2023-07-08 jrick struct got_reference *head_ref = NULL, *head_ref2 = NULL;
2926 3d1b16d1 2023-07-08 jrick struct got_reference *commit_ref = NULL;
2927 3d1b16d1 2023-07-08 jrick struct got_object_id *head_commit_id = NULL;
2928 3d1b16d1 2023-07-08 jrick struct got_object_id *head_commit_id2 = NULL;
2929 3d1b16d1 2023-07-08 jrick char *head_refname = NULL;
2930 3d1b16d1 2023-07-08 jrick char *commit_refname = NULL;
2931 3d1b16d1 2023-07-08 jrick char *diff_path = NULL;
2932 3d1b16d1 2023-07-08 jrick int have_staged_files = 0;
2933 3d1b16d1 2023-07-08 jrick int sendfd = -1;
2934 3d1b16d1 2023-07-08 jrick pid_t sendpid = -1;
2935 3d1b16d1 2023-07-08 jrick struct got_send_progress_arg spa;
2936 3d1b16d1 2023-07-08 jrick struct got_pathlist_head commit_reflist;
2937 3d1b16d1 2023-07-08 jrick struct got_pathlist_head tag_names;
2938 3d1b16d1 2023-07-08 jrick struct got_pathlist_head delete_branches;
2939 3d1b16d1 2023-07-08 jrick
2940 3d1b16d1 2023-07-08 jrick *new_commit_id = NULL;
2941 3d1b16d1 2023-07-08 jrick
2942 3d1b16d1 2023-07-08 jrick memset(&cc_arg, 0, sizeof(cc_arg));
2943 3d1b16d1 2023-07-08 jrick TAILQ_INIT(&commitable_paths);
2944 3d1b16d1 2023-07-08 jrick TAILQ_INIT(&commit_reflist);
2945 3d1b16d1 2023-07-08 jrick TAILQ_INIT(&tag_names);
2946 3d1b16d1 2023-07-08 jrick TAILQ_INIT(&delete_branches);
2947 3d1b16d1 2023-07-08 jrick
2948 3d1b16d1 2023-07-08 jrick err = lock_worktree(worktree, LOCK_EX);
2949 3d1b16d1 2023-07-08 jrick if (err)
2950 3d1b16d1 2023-07-08 jrick goto done;
2951 3d1b16d1 2023-07-08 jrick
2952 3d1b16d1 2023-07-08 jrick err = got_worktree_cvg_get_commit_ref_name(&commit_refname,
2953 3d1b16d1 2023-07-08 jrick worktree);
2954 3d1b16d1 2023-07-08 jrick if (err)
2955 3d1b16d1 2023-07-08 jrick goto done;
2956 3d1b16d1 2023-07-08 jrick
2957 3d1b16d1 2023-07-08 jrick head_refname = worktree->head_ref_name;
2958 3d1b16d1 2023-07-08 jrick err = got_ref_open(&head_ref, repo, head_refname, 0);
2959 3d1b16d1 2023-07-08 jrick if (err)
2960 3d1b16d1 2023-07-08 jrick goto done;
2961 3d1b16d1 2023-07-08 jrick err = got_ref_resolve(&head_commit_id, repo, head_ref);
2962 3d1b16d1 2023-07-08 jrick if (err)
2963 3d1b16d1 2023-07-08 jrick goto done;
2964 3d1b16d1 2023-07-08 jrick
2965 3d1b16d1 2023-07-08 jrick err = got_ref_alloc(&commit_ref, commit_refname, head_commit_id);
2966 3d1b16d1 2023-07-08 jrick if (err)
2967 3d1b16d1 2023-07-08 jrick goto done;
2968 3d1b16d1 2023-07-08 jrick err = got_ref_write(commit_ref, repo);
2969 3d1b16d1 2023-07-08 jrick if (err)
2970 3d1b16d1 2023-07-08 jrick goto done;
2971 3d1b16d1 2023-07-08 jrick
2972 3d1b16d1 2023-07-08 jrick err = open_fileindex(&fileindex, &fileindex_path, worktree);
2973 3d1b16d1 2023-07-08 jrick if (err)
2974 3d1b16d1 2023-07-08 jrick goto done;
2975 3d1b16d1 2023-07-08 jrick
2976 3d1b16d1 2023-07-08 jrick err = got_fileindex_for_each_entry_safe(fileindex, check_staged_file,
2977 3d1b16d1 2023-07-08 jrick &have_staged_files);
2978 3d1b16d1 2023-07-08 jrick if (err && err->code != GOT_ERR_CANCELLED)
2979 3d1b16d1 2023-07-08 jrick goto done;
2980 3d1b16d1 2023-07-08 jrick if (have_staged_files) {
2981 3d1b16d1 2023-07-08 jrick err = check_non_staged_files(fileindex, paths);
2982 3d1b16d1 2023-07-08 jrick if (err)
2983 3d1b16d1 2023-07-08 jrick goto done;
2984 3d1b16d1 2023-07-08 jrick }
2985 3d1b16d1 2023-07-08 jrick
2986 3d1b16d1 2023-07-08 jrick cc_arg.commitable_paths = &commitable_paths;
2987 3d1b16d1 2023-07-08 jrick cc_arg.worktree = worktree;
2988 3d1b16d1 2023-07-08 jrick cc_arg.fileindex = fileindex;
2989 3d1b16d1 2023-07-08 jrick cc_arg.repo = repo;
2990 3d1b16d1 2023-07-08 jrick cc_arg.have_staged_files = have_staged_files;
2991 3d1b16d1 2023-07-08 jrick cc_arg.allow_bad_symlinks = allow_bad_symlinks;
2992 3d1b16d1 2023-07-08 jrick cc_arg.diff_header_shown = 0;
2993 3d1b16d1 2023-07-08 jrick cc_arg.commit_conflicts = commit_conflicts;
2994 3d1b16d1 2023-07-08 jrick if (show_diff) {
2995 3d1b16d1 2023-07-08 jrick err = got_opentemp_named(&diff_path, &cc_arg.diff_outfile,
2996 3d1b16d1 2023-07-08 jrick GOT_TMPDIR_STR "/got", ".diff");
2997 3d1b16d1 2023-07-08 jrick if (err)
2998 3d1b16d1 2023-07-08 jrick goto done;
2999 3d1b16d1 2023-07-08 jrick cc_arg.f1 = got_opentemp();
3000 3d1b16d1 2023-07-08 jrick if (cc_arg.f1 == NULL) {
3001 3d1b16d1 2023-07-08 jrick err = got_error_from_errno("got_opentemp");
3002 3d1b16d1 2023-07-08 jrick goto done;
3003 3d1b16d1 2023-07-08 jrick }
3004 3d1b16d1 2023-07-08 jrick cc_arg.f2 = got_opentemp();
3005 3d1b16d1 2023-07-08 jrick if (cc_arg.f2 == NULL) {
3006 3d1b16d1 2023-07-08 jrick err = got_error_from_errno("got_opentemp");
3007 3d1b16d1 2023-07-08 jrick goto done;
3008 3d1b16d1 2023-07-08 jrick }
3009 3d1b16d1 2023-07-08 jrick }
3010 3d1b16d1 2023-07-08 jrick
3011 3d1b16d1 2023-07-08 jrick TAILQ_FOREACH(pe, paths, entry) {
3012 3d1b16d1 2023-07-08 jrick err = worktree_status(worktree, pe->path, fileindex, repo,
3013 3d1b16d1 2023-07-08 jrick collect_commitables, &cc_arg, NULL, NULL, 0, 0);
3014 3d1b16d1 2023-07-08 jrick if (err)
3015 3d1b16d1 2023-07-08 jrick goto done;
3016 3d1b16d1 2023-07-08 jrick }
3017 3d1b16d1 2023-07-08 jrick
3018 3d1b16d1 2023-07-08 jrick if (show_diff) {
3019 3d1b16d1 2023-07-08 jrick if (fflush(cc_arg.diff_outfile) == EOF) {
3020 3d1b16d1 2023-07-08 jrick err = got_error_from_errno("fflush");
3021 3d1b16d1 2023-07-08 jrick goto done;
3022 3d1b16d1 2023-07-08 jrick }
3023 3d1b16d1 2023-07-08 jrick }
3024 3d1b16d1 2023-07-08 jrick
3025 3d1b16d1 2023-07-08 jrick if (TAILQ_EMPTY(&commitable_paths)) {
3026 3d1b16d1 2023-07-08 jrick err = got_error(GOT_ERR_COMMIT_NO_CHANGES);
3027 3d1b16d1 2023-07-08 jrick goto done;
3028 3d1b16d1 2023-07-08 jrick }
3029 3d1b16d1 2023-07-08 jrick
3030 3d1b16d1 2023-07-08 jrick TAILQ_FOREACH(pe, paths, entry) {
3031 3d1b16d1 2023-07-08 jrick err = check_path_is_commitable(pe->path, &commitable_paths);
3032 3d1b16d1 2023-07-08 jrick if (err)
3033 3d1b16d1 2023-07-08 jrick goto done;
3034 3d1b16d1 2023-07-08 jrick }
3035 3d1b16d1 2023-07-08 jrick
3036 3d1b16d1 2023-07-08 jrick TAILQ_FOREACH(pe, &commitable_paths, entry) {
3037 3d1b16d1 2023-07-08 jrick struct got_commitable *ct = pe->data;
3038 3d1b16d1 2023-07-08 jrick const char *ct_path = ct->in_repo_path;
3039 3d1b16d1 2023-07-08 jrick
3040 3d1b16d1 2023-07-08 jrick while (ct_path[0] == '/')
3041 3d1b16d1 2023-07-08 jrick ct_path++;
3042 3d1b16d1 2023-07-08 jrick err = check_out_of_date(ct_path, ct->status,
3043 3d1b16d1 2023-07-08 jrick ct->staged_status, ct->base_blob_id, ct->base_commit_id,
3044 3d1b16d1 2023-07-08 jrick head_commit_id, repo, GOT_ERR_COMMIT_OUT_OF_DATE);
3045 3d1b16d1 2023-07-08 jrick if (err)
3046 3d1b16d1 2023-07-08 jrick goto done;
3047 3d1b16d1 2023-07-08 jrick }
3048 3d1b16d1 2023-07-08 jrick
3049 3d1b16d1 2023-07-08 jrick err = commit_worktree(new_commit_id, &commitable_paths,
3050 3d1b16d1 2023-07-08 jrick head_commit_id, NULL, worktree, author, committer,
3051 3d1b16d1 2023-07-08 jrick (diff_path && cc_arg.diff_header_shown) ? diff_path : NULL,
3052 3d1b16d1 2023-07-08 jrick commit_msg_cb, commit_arg, status_cb, status_arg, repo);
3053 3d1b16d1 2023-07-08 jrick if (err)
3054 3d1b16d1 2023-07-08 jrick goto done;
3055 3d1b16d1 2023-07-08 jrick
3056 3d1b16d1 2023-07-08 jrick /*
3057 3d1b16d1 2023-07-08 jrick * Check if a concurrent commit to our branch has occurred.
3058 3d1b16d1 2023-07-08 jrick * Lock the reference here to prevent concurrent modification.
3059 3d1b16d1 2023-07-08 jrick */
3060 3d1b16d1 2023-07-08 jrick err = got_ref_open(&head_ref2, repo, head_refname, 1);
3061 3d1b16d1 2023-07-08 jrick if (err)
3062 3d1b16d1 2023-07-08 jrick goto done;
3063 3d1b16d1 2023-07-08 jrick err = got_ref_resolve(&head_commit_id2, repo, head_ref2);
3064 3d1b16d1 2023-07-08 jrick if (err)
3065 3d1b16d1 2023-07-08 jrick goto done;
3066 3d1b16d1 2023-07-08 jrick if (got_object_id_cmp(head_commit_id, head_commit_id2) != 0) {
3067 3d1b16d1 2023-07-08 jrick err = got_error(GOT_ERR_COMMIT_HEAD_CHANGED);
3068 3d1b16d1 2023-07-08 jrick goto done;
3069 3d1b16d1 2023-07-08 jrick }
3070 3d1b16d1 2023-07-08 jrick
3071 3d1b16d1 2023-07-08 jrick err = got_pathlist_append(&commit_reflist, commit_refname,
3072 3d1b16d1 2023-07-08 jrick head_refname);
3073 3d1b16d1 2023-07-08 jrick if (err)
3074 3d1b16d1 2023-07-08 jrick goto done;
3075 3d1b16d1 2023-07-08 jrick
3076 3d1b16d1 2023-07-08 jrick /* Update commit ref in repository. */
3077 3d1b16d1 2023-07-08 jrick err = got_ref_change_ref(commit_ref, *new_commit_id);
3078 3d1b16d1 2023-07-08 jrick if (err)
3079 3d1b16d1 2023-07-08 jrick goto done;
3080 3d1b16d1 2023-07-08 jrick err = got_ref_write(commit_ref, repo);
3081 3d1b16d1 2023-07-08 jrick if (err)
3082 3d1b16d1 2023-07-08 jrick goto done;
3083 3d1b16d1 2023-07-08 jrick
3084 3d1b16d1 2023-07-08 jrick if (verbosity >= 0) {
3085 3d1b16d1 2023-07-08 jrick printf("Connecting to \"%s\" %s://%s%s%s%s%s\n",
3086 3d1b16d1 2023-07-08 jrick remote->name, proto, host,
3087 3d1b16d1 2023-07-08 jrick port ? ":" : "", port ? port : "",
3088 3d1b16d1 2023-07-08 jrick *server_path == '/' ? "" : "/", server_path);
3089 3d1b16d1 2023-07-08 jrick }
3090 3d1b16d1 2023-07-08 jrick
3091 3d1b16d1 2023-07-08 jrick /* Attempt send to remote branch. */
3092 3d1b16d1 2023-07-08 jrick err = got_send_connect(&sendpid, &sendfd, proto, host, port,
3093 3d1b16d1 2023-07-08 jrick server_path, verbosity);
3094 3d1b16d1 2023-07-08 jrick if (err)
3095 3d1b16d1 2023-07-08 jrick goto done;
3096 3d1b16d1 2023-07-08 jrick
3097 3d1b16d1 2023-07-08 jrick memset(&spa, 0, sizeof(spa));
3098 3d1b16d1 2023-07-08 jrick spa.last_scaled_packsize[0] = '\0';
3099 3d1b16d1 2023-07-08 jrick spa.last_p_deltify = -1;
3100 3d1b16d1 2023-07-08 jrick spa.last_p_written = -1;
3101 3d1b16d1 2023-07-08 jrick spa.verbosity = verbosity;
3102 3d1b16d1 2023-07-08 jrick spa.delete_branches = &delete_branches;
3103 3d1b16d1 2023-07-08 jrick err = got_send_pack(remote->name, &commit_reflist, &tag_names,
3104 3d1b16d1 2023-07-08 jrick &delete_branches, verbosity, 0, sendfd, repo, send_progress, &spa,
3105 3d1b16d1 2023-07-08 jrick check_cancelled, NULL);
3106 3d1b16d1 2023-07-08 jrick if (spa.printed_something)
3107 3d1b16d1 2023-07-08 jrick putchar('\n');
3108 3d1b16d1 2023-07-08 jrick if (err != NULL && err->code == GOT_ERR_SEND_ANCESTRY) {
3109 3d1b16d1 2023-07-08 jrick /*
3110 3d1b16d1 2023-07-08 jrick * Fetch new changes since remote has diverged.
3111 3d1b16d1 2023-07-08 jrick * No trivial-rebase yet; require update to be run manually.
3112 3d1b16d1 2023-07-08 jrick */
3113 3d1b16d1 2023-07-08 jrick err = fetch_updated_remote(proto, host, port, server_path,
3114 3d1b16d1 2023-07-08 jrick verbosity, remote, repo, head_ref, head_refname);
3115 3d1b16d1 2023-07-08 jrick if (err == NULL)
3116 3d1b16d1 2023-07-08 jrick goto done;
3117 3d1b16d1 2023-07-08 jrick err = got_error(GOT_ERR_COMMIT_OUT_OF_DATE);
3118 3d1b16d1 2023-07-08 jrick goto done;
3119 3d1b16d1 2023-07-08 jrick /* XXX: Rebase commit over fetched remote branch. */
3120 3d1b16d1 2023-07-08 jrick }
3121 3d1b16d1 2023-07-08 jrick if (err) {
3122 3d1b16d1 2023-07-08 jrick goto done;
3123 3d1b16d1 2023-07-08 jrick }
3124 3d1b16d1 2023-07-08 jrick
3125 3d1b16d1 2023-07-08 jrick /* Update branch head in repository. */
3126 3d1b16d1 2023-07-08 jrick err = got_ref_change_ref(head_ref2, *new_commit_id);
3127 3d1b16d1 2023-07-08 jrick if (err)
3128 3d1b16d1 2023-07-08 jrick goto done;
3129 3d1b16d1 2023-07-08 jrick err = got_ref_write(head_ref2, repo);
3130 3d1b16d1 2023-07-08 jrick if (err)
3131 3d1b16d1 2023-07-08 jrick goto done;
3132 3d1b16d1 2023-07-08 jrick
3133 3d1b16d1 2023-07-08 jrick err = got_worktree_set_base_commit_id(worktree, repo, *new_commit_id);
3134 3d1b16d1 2023-07-08 jrick if (err)
3135 3d1b16d1 2023-07-08 jrick goto done;
3136 3d1b16d1 2023-07-08 jrick
3137 3d1b16d1 2023-07-08 jrick err = ref_base_commit(worktree, repo);
3138 3d1b16d1 2023-07-08 jrick if (err)
3139 3d1b16d1 2023-07-08 jrick goto done;
3140 3d1b16d1 2023-07-08 jrick
3141 3d1b16d1 2023-07-08 jrick /* XXX: fileindex must be updated for other fetched changes? */
3142 3d1b16d1 2023-07-08 jrick err = update_fileindex_after_commit(worktree, &commitable_paths,
3143 3d1b16d1 2023-07-08 jrick *new_commit_id, fileindex, have_staged_files);
3144 3d1b16d1 2023-07-08 jrick sync_err = sync_fileindex(fileindex, fileindex_path);
3145 3d1b16d1 2023-07-08 jrick if (sync_err && err == NULL)
3146 3d1b16d1 2023-07-08 jrick err = sync_err;
3147 3d1b16d1 2023-07-08 jrick done:
3148 3d1b16d1 2023-07-08 jrick if (head_ref2) {
3149 3d1b16d1 2023-07-08 jrick unlockerr = got_ref_unlock(head_ref2);
3150 3d1b16d1 2023-07-08 jrick if (unlockerr && err == NULL)
3151 3d1b16d1 2023-07-08 jrick err = unlockerr;
3152 3d1b16d1 2023-07-08 jrick got_ref_close(head_ref2);
3153 3d1b16d1 2023-07-08 jrick }
3154 3d1b16d1 2023-07-08 jrick if (commit_ref)
3155 3d1b16d1 2023-07-08 jrick got_ref_close(commit_ref);
3156 3d1b16d1 2023-07-08 jrick if (fileindex)
3157 3d1b16d1 2023-07-08 jrick got_fileindex_free(fileindex);
3158 3d1b16d1 2023-07-08 jrick unlockerr = lock_worktree(worktree, LOCK_SH);
3159 3d1b16d1 2023-07-08 jrick if (unlockerr && err == NULL)
3160 3d1b16d1 2023-07-08 jrick err = unlockerr;
3161 3d1b16d1 2023-07-08 jrick TAILQ_FOREACH(pe, &commitable_paths, entry) {
3162 3d1b16d1 2023-07-08 jrick struct got_commitable *ct = pe->data;
3163 3d1b16d1 2023-07-08 jrick
3164 3d1b16d1 2023-07-08 jrick free_commitable(ct);
3165 3d1b16d1 2023-07-08 jrick }
3166 3d1b16d1 2023-07-08 jrick got_pathlist_free(&commitable_paths, GOT_PATHLIST_FREE_NONE);
3167 3d1b16d1 2023-07-08 jrick if (diff_path && unlink(diff_path) == -1 && err == NULL)
3168 3d1b16d1 2023-07-08 jrick err = got_error_from_errno2("unlink", diff_path);
3169 3d1b16d1 2023-07-08 jrick if (cc_arg.diff_outfile && fclose(cc_arg.diff_outfile) == EOF &&
3170 3d1b16d1 2023-07-08 jrick err == NULL)
3171 3d1b16d1 2023-07-08 jrick err = got_error_from_errno("fclose");
3172 3d1b16d1 2023-07-08 jrick free(head_commit_id);
3173 3d1b16d1 2023-07-08 jrick free(head_commit_id2);
3174 3d1b16d1 2023-07-08 jrick free(commit_refname);
3175 3d1b16d1 2023-07-08 jrick free(fileindex_path);
3176 3d1b16d1 2023-07-08 jrick free(diff_path);
3177 3d1b16d1 2023-07-08 jrick return err;
3178 3d1b16d1 2023-07-08 jrick }
3179 3d1b16d1 2023-07-08 jrick
3180 3d1b16d1 2023-07-08 jrick const struct got_error *
3181 3d1b16d1 2023-07-08 jrick got_worktree_cvg_get_commit_ref_name(char **refname,
3182 3d1b16d1 2023-07-08 jrick struct got_worktree *worktree)
3183 3d1b16d1 2023-07-08 jrick {
3184 3d1b16d1 2023-07-08 jrick return get_ref_name(refname, worktree, GOT_WORKTREE_COMMIT_REF_PREFIX);
3185 3d1b16d1 2023-07-08 jrick }