James Bowes | 6757ada | 2007-03-13 21:58:22 -0400 | [diff] [blame] | 1 | /* |
| 2 | * git gc builtin command |
| 3 | * |
| 4 | * Cleanup unreachable files and optimize the repository. |
| 5 | * |
| 6 | * Copyright (c) 2007 James Bowes |
| 7 | * |
| 8 | * Based on git-gc.sh, which is |
| 9 | * |
| 10 | * Copyright (c) 2006 Shawn O. Pearce |
| 11 | */ |
| 12 | |
Peter Hagervall | baffc0e | 2007-07-15 01:14:45 +0200 | [diff] [blame] | 13 | #include "builtin.h" |
Stefan Beller | a80d72d | 2018-03-23 18:20:59 +0100 | [diff] [blame] | 14 | #include "repository.h" |
Brandon Williams | b2141fc | 2017-06-14 11:07:36 -0700 | [diff] [blame] | 15 | #include "config.h" |
Michael Haggerty | ebebeae | 2015-08-10 11:47:49 +0200 | [diff] [blame] | 16 | #include "tempfile.h" |
Michael Haggerty | 697cc8e | 2014-10-01 12:28:42 +0200 | [diff] [blame] | 17 | #include "lockfile.h" |
James Bowes | 44c637c | 2007-11-01 21:02:27 -0400 | [diff] [blame] | 18 | #include "parse-options.h" |
James Bowes | 6757ada | 2007-03-13 21:58:22 -0400 | [diff] [blame] | 19 | #include "run-command.h" |
Jonathan Nieder | 4c5baf0 | 2013-10-16 16:11:46 -0700 | [diff] [blame] | 20 | #include "sigchain.h" |
Jeff King | dbbcd44 | 2020-07-28 16:23:39 -0400 | [diff] [blame] | 21 | #include "strvec.h" |
Nguyễn Thái Ngọc Duy | eab3296 | 2013-12-05 20:02:54 +0700 | [diff] [blame] | 22 | #include "commit.h" |
Derrick Stolee | d5d5d7b | 2018-06-27 09:24:46 -0400 | [diff] [blame] | 23 | #include "commit-graph.h" |
Jonathan Tan | 0abe14f | 2017-08-18 15:20:26 -0700 | [diff] [blame] | 24 | #include "packfile.h" |
Stefan Beller | a80d72d | 2018-03-23 18:20:59 +0100 | [diff] [blame] | 25 | #include "object-store.h" |
Nguyễn Thái Ngọc Duy | 9806f5a | 2018-04-15 17:36:17 +0200 | [diff] [blame] | 26 | #include "pack.h" |
| 27 | #include "pack-objects.h" |
| 28 | #include "blob.h" |
| 29 | #include "tree.h" |
Christian Couder | b14ed5a | 2019-06-25 15:40:31 +0200 | [diff] [blame] | 30 | #include "promisor-remote.h" |
Derrick Stolee | 4ddc79b | 2020-09-17 18:11:51 +0000 | [diff] [blame] | 31 | #include "refs.h" |
Derrick Stolee | 28cb5e6 | 2020-09-25 12:33:31 +0000 | [diff] [blame] | 32 | #include "remote.h" |
Derrick Stolee | e841a79 | 2020-09-25 12:33:38 +0000 | [diff] [blame] | 33 | #include "object-store.h" |
Derrick Stolee | 2fec604 | 2020-09-11 17:49:18 +0000 | [diff] [blame] | 34 | #include "exec-cmd.h" |
James Bowes | 6757ada | 2007-03-13 21:58:22 -0400 | [diff] [blame] | 35 | |
| 36 | #define FAILED_RUN "failed to run %s" |
| 37 | |
James Bowes | 44c637c | 2007-11-01 21:02:27 -0400 | [diff] [blame] | 38 | static const char * const builtin_gc_usage[] = { |
Alex Henrie | 9c9b4f2 | 2015-01-13 00:44:47 -0700 | [diff] [blame] | 39 | N_("git gc [<options>]"), |
James Bowes | 44c637c | 2007-11-01 21:02:27 -0400 | [diff] [blame] | 40 | NULL |
| 41 | }; |
James Bowes | 6757ada | 2007-03-13 21:58:22 -0400 | [diff] [blame] | 42 | |
Linus Torvalds | 5675239 | 2007-05-24 11:41:39 -0700 | [diff] [blame] | 43 | static int pack_refs = 1; |
Nguyễn Thái Ngọc Duy | 62aad18 | 2014-05-25 07:38:29 +0700 | [diff] [blame] | 44 | static int prune_reflogs = 1; |
Jeff King | 07e7dbf | 2016-08-11 12:13:09 -0400 | [diff] [blame] | 45 | static int aggressive_depth = 50; |
Johannes Schindelin | 1c192f3 | 2007-12-06 12:03:38 +0000 | [diff] [blame] | 46 | static int aggressive_window = 250; |
Junio C Hamano | 2c3c439 | 2007-09-05 13:01:37 -0700 | [diff] [blame] | 47 | static int gc_auto_threshold = 6700; |
Junio C Hamano | 9706397 | 2008-03-23 00:04:48 -0700 | [diff] [blame] | 48 | static int gc_auto_pack_limit = 50; |
Nguyễn Thái Ngọc Duy | 9f673f9 | 2014-02-08 14:08:52 +0700 | [diff] [blame] | 49 | static int detach_auto = 1; |
Johannes Schindelin | dddbad7 | 2017-04-26 21:29:31 +0200 | [diff] [blame] | 50 | static timestamp_t gc_log_expire_time; |
David Turner | a831c06 | 2017-02-10 16:28:22 -0500 | [diff] [blame] | 51 | static const char *gc_log_expire = "1.day.ago"; |
David Bryson | d3154b4 | 2008-09-30 13:28:58 -0700 | [diff] [blame] | 52 | static const char *prune_expire = "2.weeks.ago"; |
Nguyễn Thái Ngọc Duy | e3df33b | 2014-11-30 15:24:53 +0700 | [diff] [blame] | 53 | static const char *prune_worktrees_expire = "3.months.ago"; |
Nguyễn Thái Ngọc Duy | 55dfe13 | 2018-04-15 17:36:15 +0200 | [diff] [blame] | 54 | static unsigned long big_pack_threshold; |
Nguyễn Thái Ngọc Duy | 9806f5a | 2018-04-15 17:36:17 +0200 | [diff] [blame] | 55 | static unsigned long max_delta_cache_size = DEFAULT_DELTA_CACHE_SIZE; |
James Bowes | 6757ada | 2007-03-13 21:58:22 -0400 | [diff] [blame] | 56 | |
Jeff King | 22f9b7f | 2020-07-28 16:24:27 -0400 | [diff] [blame] | 57 | static struct strvec reflog = STRVEC_INIT; |
| 58 | static struct strvec repack = STRVEC_INIT; |
| 59 | static struct strvec prune = STRVEC_INIT; |
| 60 | static struct strvec prune_worktrees = STRVEC_INIT; |
| 61 | static struct strvec rerere = STRVEC_INIT; |
James Bowes | 6757ada | 2007-03-13 21:58:22 -0400 | [diff] [blame] | 62 | |
Jeff King | 076aa2c | 2017-09-05 08:15:08 -0400 | [diff] [blame] | 63 | static struct tempfile *pidfile; |
Nguyễn Thái Ngọc Duy | 329e6e8 | 2015-09-19 12:13:23 +0700 | [diff] [blame] | 64 | static struct lock_file log_lock; |
Jonathan Nieder | 4c5baf0 | 2013-10-16 16:11:46 -0700 | [diff] [blame] | 65 | |
Doug Kelly | 478f34d | 2015-11-03 21:05:08 -0600 | [diff] [blame] | 66 | static struct string_list pack_garbage = STRING_LIST_INIT_DUP; |
| 67 | |
| 68 | static void clean_pack_garbage(void) |
| 69 | { |
| 70 | int i; |
| 71 | for (i = 0; i < pack_garbage.nr; i++) |
| 72 | unlink_or_warn(pack_garbage.items[i].string); |
| 73 | string_list_clear(&pack_garbage, 0); |
| 74 | } |
| 75 | |
| 76 | static void report_pack_garbage(unsigned seen_bits, const char *path) |
| 77 | { |
| 78 | if (seen_bits == PACKDIR_FILE_IDX) |
| 79 | string_list_append(&pack_garbage, path); |
| 80 | } |
| 81 | |
Nguyễn Thái Ngọc Duy | 329e6e8 | 2015-09-19 12:13:23 +0700 | [diff] [blame] | 82 | static void process_log_file(void) |
| 83 | { |
| 84 | struct stat st; |
David Turner | a831c06 | 2017-02-10 16:28:22 -0500 | [diff] [blame] | 85 | if (fstat(get_lock_file_fd(&log_lock), &st)) { |
| 86 | /* |
| 87 | * Perhaps there was an i/o error or another |
| 88 | * unlikely situation. Try to make a note of |
| 89 | * this in gc.log along with any existing |
| 90 | * messages. |
| 91 | */ |
| 92 | int saved_errno = errno; |
| 93 | fprintf(stderr, _("Failed to fstat %s: %s"), |
Martin Ågren | d4a4976 | 2021-01-05 20:23:46 +0100 | [diff] [blame] | 94 | get_lock_file_path(&log_lock), |
David Turner | a831c06 | 2017-02-10 16:28:22 -0500 | [diff] [blame] | 95 | strerror(saved_errno)); |
| 96 | fflush(stderr); |
Nguyễn Thái Ngọc Duy | 329e6e8 | 2015-09-19 12:13:23 +0700 | [diff] [blame] | 97 | commit_lock_file(&log_lock); |
David Turner | a831c06 | 2017-02-10 16:28:22 -0500 | [diff] [blame] | 98 | errno = saved_errno; |
| 99 | } else if (st.st_size) { |
| 100 | /* There was some error recorded in the lock file */ |
| 101 | commit_lock_file(&log_lock); |
| 102 | } else { |
| 103 | /* No error, clean up any old gc.log */ |
| 104 | unlink(git_path("gc.log")); |
Nguyễn Thái Ngọc Duy | 329e6e8 | 2015-09-19 12:13:23 +0700 | [diff] [blame] | 105 | rollback_lock_file(&log_lock); |
David Turner | a831c06 | 2017-02-10 16:28:22 -0500 | [diff] [blame] | 106 | } |
Nguyễn Thái Ngọc Duy | 329e6e8 | 2015-09-19 12:13:23 +0700 | [diff] [blame] | 107 | } |
| 108 | |
| 109 | static void process_log_file_at_exit(void) |
| 110 | { |
| 111 | fflush(stderr); |
| 112 | process_log_file(); |
| 113 | } |
| 114 | |
| 115 | static void process_log_file_on_signal(int signo) |
| 116 | { |
| 117 | process_log_file(); |
| 118 | sigchain_pop(signo); |
| 119 | raise(signo); |
| 120 | } |
| 121 | |
Ævar Arnfjörð Bjarmason | bf3d70f | 2019-03-28 17:14:34 +0100 | [diff] [blame] | 122 | static int gc_config_is_timestamp_never(const char *var) |
| 123 | { |
| 124 | const char *value; |
| 125 | timestamp_t expire; |
| 126 | |
| 127 | if (!git_config_get_value(var, &value) && value) { |
| 128 | if (parse_expiry_date(value, &expire)) |
| 129 | die(_("failed to parse '%s' value '%s'"), var, value); |
| 130 | return expire == 0; |
| 131 | } |
| 132 | return 0; |
| 133 | } |
| 134 | |
Tanay Abhra | 5801d3b | 2014-08-07 09:21:22 -0700 | [diff] [blame] | 135 | static void gc_config(void) |
James Bowes | 6757ada | 2007-03-13 21:58:22 -0400 | [diff] [blame] | 136 | { |
Tanay Abhra | 5801d3b | 2014-08-07 09:21:22 -0700 | [diff] [blame] | 137 | const char *value; |
| 138 | |
| 139 | if (!git_config_get_value("gc.packrefs", &value)) { |
Miklos Vajna | c5e5a2c | 2008-02-08 15:26:18 +0100 | [diff] [blame] | 140 | if (value && !strcmp(value, "notbare")) |
James Bowes | 6757ada | 2007-03-13 21:58:22 -0400 | [diff] [blame] | 141 | pack_refs = -1; |
| 142 | else |
Tanay Abhra | 5801d3b | 2014-08-07 09:21:22 -0700 | [diff] [blame] | 143 | pack_refs = git_config_bool("gc.packrefs", value); |
James Bowes | 6757ada | 2007-03-13 21:58:22 -0400 | [diff] [blame] | 144 | } |
Tanay Abhra | 5801d3b | 2014-08-07 09:21:22 -0700 | [diff] [blame] | 145 | |
Ævar Arnfjörð Bjarmason | bf3d70f | 2019-03-28 17:14:34 +0100 | [diff] [blame] | 146 | if (gc_config_is_timestamp_never("gc.reflogexpire") && |
| 147 | gc_config_is_timestamp_never("gc.reflogexpireunreachable")) |
| 148 | prune_reflogs = 0; |
| 149 | |
Tanay Abhra | 5801d3b | 2014-08-07 09:21:22 -0700 | [diff] [blame] | 150 | git_config_get_int("gc.aggressivewindow", &aggressive_window); |
| 151 | git_config_get_int("gc.aggressivedepth", &aggressive_depth); |
| 152 | git_config_get_int("gc.auto", &gc_auto_threshold); |
| 153 | git_config_get_int("gc.autopacklimit", &gc_auto_pack_limit); |
| 154 | git_config_get_bool("gc.autodetach", &detach_auto); |
Christian Couder | 77d6797 | 2017-02-27 19:00:13 +0100 | [diff] [blame] | 155 | git_config_get_expiry("gc.pruneexpire", &prune_expire); |
| 156 | git_config_get_expiry("gc.worktreepruneexpire", &prune_worktrees_expire); |
Junio C Hamano | 94c9b5a | 2017-03-17 13:50:23 -0700 | [diff] [blame] | 157 | git_config_get_expiry("gc.logexpiry", &gc_log_expire); |
David Turner | a831c06 | 2017-02-10 16:28:22 -0500 | [diff] [blame] | 158 | |
Nguyễn Thái Ngọc Duy | 55dfe13 | 2018-04-15 17:36:15 +0200 | [diff] [blame] | 159 | git_config_get_ulong("gc.bigpackthreshold", &big_pack_threshold); |
Nguyễn Thái Ngọc Duy | 9806f5a | 2018-04-15 17:36:17 +0200 | [diff] [blame] | 160 | git_config_get_ulong("pack.deltacachesize", &max_delta_cache_size); |
Nguyễn Thái Ngọc Duy | 55dfe13 | 2018-04-15 17:36:15 +0200 | [diff] [blame] | 161 | |
Tanay Abhra | 5801d3b | 2014-08-07 09:21:22 -0700 | [diff] [blame] | 162 | git_config(git_default_config, NULL); |
James Bowes | 6757ada | 2007-03-13 21:58:22 -0400 | [diff] [blame] | 163 | } |
| 164 | |
Derrick Stolee | 41abfe1 | 2021-02-09 13:42:28 +0000 | [diff] [blame] | 165 | struct maintenance_run_opts; |
| 166 | static int maintenance_task_pack_refs(MAYBE_UNUSED struct maintenance_run_opts *opts) |
| 167 | { |
| 168 | struct strvec pack_refs_cmd = STRVEC_INIT; |
| 169 | strvec_pushl(&pack_refs_cmd, "pack-refs", "--all", "--prune", NULL); |
| 170 | |
| 171 | return run_command_v_opt(pack_refs_cmd.v, RUN_GIT_CMD); |
| 172 | } |
| 173 | |
Junio C Hamano | a087cc9 | 2007-09-17 00:44:17 -0700 | [diff] [blame] | 174 | static int too_many_loose_objects(void) |
Junio C Hamano | 2c3c439 | 2007-09-05 13:01:37 -0700 | [diff] [blame] | 175 | { |
| 176 | /* |
| 177 | * Quickly check if a "gc" is needed, by estimating how |
| 178 | * many loose objects there are. Because SHA-1 is evenly |
| 179 | * distributed, we can check only one and get a reasonable |
| 180 | * estimate. |
| 181 | */ |
Junio C Hamano | 2c3c439 | 2007-09-05 13:01:37 -0700 | [diff] [blame] | 182 | DIR *dir; |
| 183 | struct dirent *ent; |
| 184 | int auto_threshold; |
| 185 | int num_loose = 0; |
| 186 | int needed = 0; |
Ævar Arnfjörð Bjarmason | e5cdbd5 | 2019-03-15 16:59:53 +0100 | [diff] [blame] | 187 | const unsigned hexsz_loose = the_hash_algo->hexsz - 2; |
Junio C Hamano | 2c3c439 | 2007-09-05 13:01:37 -0700 | [diff] [blame] | 188 | |
Jeff King | 07af889 | 2017-03-28 15:47:03 -0400 | [diff] [blame] | 189 | dir = opendir(git_path("objects/17")); |
Junio C Hamano | 2c3c439 | 2007-09-05 13:01:37 -0700 | [diff] [blame] | 190 | if (!dir) |
| 191 | return 0; |
| 192 | |
René Scharfe | 42c78a2 | 2017-07-08 12:35:35 +0200 | [diff] [blame] | 193 | auto_threshold = DIV_ROUND_UP(gc_auto_threshold, 256); |
Junio C Hamano | 2c3c439 | 2007-09-05 13:01:37 -0700 | [diff] [blame] | 194 | while ((ent = readdir(dir)) != NULL) { |
Ævar Arnfjörð Bjarmason | e5cdbd5 | 2019-03-15 16:59:53 +0100 | [diff] [blame] | 195 | if (strspn(ent->d_name, "0123456789abcdef") != hexsz_loose || |
| 196 | ent->d_name[hexsz_loose] != '\0') |
Junio C Hamano | 2c3c439 | 2007-09-05 13:01:37 -0700 | [diff] [blame] | 197 | continue; |
| 198 | if (++num_loose > auto_threshold) { |
| 199 | needed = 1; |
| 200 | break; |
| 201 | } |
| 202 | } |
| 203 | closedir(dir); |
| 204 | return needed; |
| 205 | } |
| 206 | |
Nguyễn Thái Ngọc Duy | 9806f5a | 2018-04-15 17:36:17 +0200 | [diff] [blame] | 207 | static struct packed_git *find_base_packs(struct string_list *packs, |
| 208 | unsigned long limit) |
Nguyễn Thái Ngọc Duy | ae4e89e | 2018-04-15 17:36:14 +0200 | [diff] [blame] | 209 | { |
| 210 | struct packed_git *p, *base = NULL; |
| 211 | |
Derrick Stolee | 454ea2e | 2018-08-20 16:52:04 +0000 | [diff] [blame] | 212 | for (p = get_all_packs(the_repository); p; p = p->next) { |
Nguyễn Thái Ngọc Duy | ae4e89e | 2018-04-15 17:36:14 +0200 | [diff] [blame] | 213 | if (!p->pack_local) |
| 214 | continue; |
Nguyễn Thái Ngọc Duy | 55dfe13 | 2018-04-15 17:36:15 +0200 | [diff] [blame] | 215 | if (limit) { |
| 216 | if (p->pack_size >= limit) |
| 217 | string_list_append(packs, p->pack_name); |
| 218 | } else if (!base || base->pack_size < p->pack_size) { |
Nguyễn Thái Ngọc Duy | ae4e89e | 2018-04-15 17:36:14 +0200 | [diff] [blame] | 219 | base = p; |
| 220 | } |
| 221 | } |
| 222 | |
| 223 | if (base) |
| 224 | string_list_append(packs, base->pack_name); |
Nguyễn Thái Ngọc Duy | 9806f5a | 2018-04-15 17:36:17 +0200 | [diff] [blame] | 225 | |
| 226 | return base; |
Nguyễn Thái Ngọc Duy | ae4e89e | 2018-04-15 17:36:14 +0200 | [diff] [blame] | 227 | } |
| 228 | |
Junio C Hamano | 1781550 | 2007-09-17 00:55:13 -0700 | [diff] [blame] | 229 | static int too_many_packs(void) |
| 230 | { |
| 231 | struct packed_git *p; |
| 232 | int cnt; |
| 233 | |
| 234 | if (gc_auto_pack_limit <= 0) |
| 235 | return 0; |
| 236 | |
Derrick Stolee | 454ea2e | 2018-08-20 16:52:04 +0000 | [diff] [blame] | 237 | for (cnt = 0, p = get_all_packs(the_repository); p; p = p->next) { |
Junio C Hamano | 1781550 | 2007-09-17 00:55:13 -0700 | [diff] [blame] | 238 | if (!p->pack_local) |
| 239 | continue; |
Brandon Casey | 01af249 | 2008-11-12 11:59:07 -0600 | [diff] [blame] | 240 | if (p->pack_keep) |
Junio C Hamano | 1781550 | 2007-09-17 00:55:13 -0700 | [diff] [blame] | 241 | continue; |
| 242 | /* |
| 243 | * Perhaps check the size of the pack and count only |
| 244 | * very small ones here? |
| 245 | */ |
| 246 | cnt++; |
| 247 | } |
Eric Wong | 5f4e3bf | 2016-06-25 06:46:47 +0000 | [diff] [blame] | 248 | return gc_auto_pack_limit < cnt; |
Junio C Hamano | 1781550 | 2007-09-17 00:55:13 -0700 | [diff] [blame] | 249 | } |
| 250 | |
Nguyễn Thái Ngọc Duy | 9806f5a | 2018-04-15 17:36:17 +0200 | [diff] [blame] | 251 | static uint64_t total_ram(void) |
| 252 | { |
| 253 | #if defined(HAVE_SYSINFO) |
| 254 | struct sysinfo si; |
| 255 | |
| 256 | if (!sysinfo(&si)) |
| 257 | return si.totalram; |
| 258 | #elif defined(HAVE_BSD_SYSCTL) && (defined(HW_MEMSIZE) || defined(HW_PHYSMEM)) |
| 259 | int64_t physical_memory; |
| 260 | int mib[2]; |
| 261 | size_t length; |
| 262 | |
| 263 | mib[0] = CTL_HW; |
| 264 | # if defined(HW_MEMSIZE) |
| 265 | mib[1] = HW_MEMSIZE; |
| 266 | # else |
| 267 | mib[1] = HW_PHYSMEM; |
| 268 | # endif |
| 269 | length = sizeof(int64_t); |
| 270 | if (!sysctl(mib, 2, &physical_memory, &length, NULL, 0)) |
| 271 | return physical_memory; |
| 272 | #elif defined(GIT_WINDOWS_NATIVE) |
| 273 | MEMORYSTATUSEX memInfo; |
| 274 | |
| 275 | memInfo.dwLength = sizeof(MEMORYSTATUSEX); |
| 276 | if (GlobalMemoryStatusEx(&memInfo)) |
| 277 | return memInfo.ullTotalPhys; |
| 278 | #endif |
| 279 | return 0; |
| 280 | } |
| 281 | |
| 282 | static uint64_t estimate_repack_memory(struct packed_git *pack) |
| 283 | { |
| 284 | unsigned long nr_objects = approximate_object_count(); |
| 285 | size_t os_cache, heap; |
| 286 | |
| 287 | if (!pack || !nr_objects) |
| 288 | return 0; |
| 289 | |
| 290 | /* |
| 291 | * First we have to scan through at least one pack. |
| 292 | * Assume enough room in OS file cache to keep the entire pack |
| 293 | * or we may accidentally evict data of other processes from |
| 294 | * the cache. |
| 295 | */ |
| 296 | os_cache = pack->pack_size + pack->index_size; |
| 297 | /* then pack-objects needs lots more for book keeping */ |
| 298 | heap = sizeof(struct object_entry) * nr_objects; |
| 299 | /* |
| 300 | * internal rev-list --all --objects takes up some memory too, |
| 301 | * let's say half of it is for blobs |
| 302 | */ |
| 303 | heap += sizeof(struct blob) * nr_objects / 2; |
| 304 | /* |
| 305 | * and the other half is for trees (commits and tags are |
| 306 | * usually insignificant) |
| 307 | */ |
| 308 | heap += sizeof(struct tree) * nr_objects / 2; |
| 309 | /* and then obj_hash[], underestimated in fact */ |
| 310 | heap += sizeof(struct object *) * nr_objects; |
| 311 | /* revindex is used also */ |
Taylor Blau | 2891b43 | 2021-01-13 17:24:54 -0500 | [diff] [blame] | 312 | heap += (sizeof(off_t) + sizeof(uint32_t)) * nr_objects; |
Nguyễn Thái Ngọc Duy | 9806f5a | 2018-04-15 17:36:17 +0200 | [diff] [blame] | 313 | /* |
| 314 | * read_sha1_file() (either at delta calculation phase, or |
| 315 | * writing phase) also fills up the delta base cache |
| 316 | */ |
| 317 | heap += delta_base_cache_limit; |
| 318 | /* and of course pack-objects has its own delta cache */ |
| 319 | heap += max_delta_cache_size; |
| 320 | |
| 321 | return os_cache + heap; |
| 322 | } |
| 323 | |
Nguyễn Thái Ngọc Duy | ae4e89e | 2018-04-15 17:36:14 +0200 | [diff] [blame] | 324 | static int keep_one_pack(struct string_list_item *item, void *data) |
| 325 | { |
Jeff King | 22f9b7f | 2020-07-28 16:24:27 -0400 | [diff] [blame] | 326 | strvec_pushf(&repack, "--keep-pack=%s", basename(item->string)); |
Nguyễn Thái Ngọc Duy | ae4e89e | 2018-04-15 17:36:14 +0200 | [diff] [blame] | 327 | return 0; |
| 328 | } |
| 329 | |
| 330 | static void add_repack_all_option(struct string_list *keep_pack) |
Jeff King | 7e52f56 | 2012-04-07 06:30:09 -0400 | [diff] [blame] | 331 | { |
| 332 | if (prune_expire && !strcmp(prune_expire, "now")) |
Jeff King | 22f9b7f | 2020-07-28 16:24:27 -0400 | [diff] [blame] | 333 | strvec_push(&repack, "-a"); |
Jeff King | 7e52f56 | 2012-04-07 06:30:09 -0400 | [diff] [blame] | 334 | else { |
Jeff King | 22f9b7f | 2020-07-28 16:24:27 -0400 | [diff] [blame] | 335 | strvec_push(&repack, "-A"); |
Jeff King | 234587f | 2012-04-18 14:10:19 -0700 | [diff] [blame] | 336 | if (prune_expire) |
Jeff King | 22f9b7f | 2020-07-28 16:24:27 -0400 | [diff] [blame] | 337 | strvec_pushf(&repack, "--unpack-unreachable=%s", prune_expire); |
Jeff King | 7e52f56 | 2012-04-07 06:30:09 -0400 | [diff] [blame] | 338 | } |
Nguyễn Thái Ngọc Duy | ae4e89e | 2018-04-15 17:36:14 +0200 | [diff] [blame] | 339 | |
| 340 | if (keep_pack) |
| 341 | for_each_string_list(keep_pack, keep_one_pack, NULL); |
Jeff King | 7e52f56 | 2012-04-07 06:30:09 -0400 | [diff] [blame] | 342 | } |
| 343 | |
David Turner | bdf56de | 2016-12-28 17:45:41 -0500 | [diff] [blame] | 344 | static void add_repack_incremental_option(void) |
| 345 | { |
Jeff King | 22f9b7f | 2020-07-28 16:24:27 -0400 | [diff] [blame] | 346 | strvec_push(&repack, "--no-write-bitmap-index"); |
David Turner | bdf56de | 2016-12-28 17:45:41 -0500 | [diff] [blame] | 347 | } |
| 348 | |
Junio C Hamano | a087cc9 | 2007-09-17 00:44:17 -0700 | [diff] [blame] | 349 | static int need_to_gc(void) |
| 350 | { |
| 351 | /* |
Brandon Casey | b14d255 | 2008-03-19 16:53:20 -0500 | [diff] [blame] | 352 | * Setting gc.auto to 0 or negative can disable the |
| 353 | * automatic gc. |
Junio C Hamano | a087cc9 | 2007-09-17 00:44:17 -0700 | [diff] [blame] | 354 | */ |
Brandon Casey | b14d255 | 2008-03-19 16:53:20 -0500 | [diff] [blame] | 355 | if (gc_auto_threshold <= 0) |
Junio C Hamano | a087cc9 | 2007-09-17 00:44:17 -0700 | [diff] [blame] | 356 | return 0; |
| 357 | |
Junio C Hamano | 1781550 | 2007-09-17 00:55:13 -0700 | [diff] [blame] | 358 | /* |
| 359 | * If there are too many loose objects, but not too many |
| 360 | * packs, we run "repack -d -l". If there are too many packs, |
| 361 | * we run "repack -A -d -l". Otherwise we tell the caller |
| 362 | * there is no need. |
| 363 | */ |
Nguyễn Thái Ngọc Duy | 55dfe13 | 2018-04-15 17:36:15 +0200 | [diff] [blame] | 364 | if (too_many_packs()) { |
| 365 | struct string_list keep_pack = STRING_LIST_INIT_NODUP; |
| 366 | |
Nguyễn Thái Ngọc Duy | 8fc6776 | 2018-04-15 17:36:16 +0200 | [diff] [blame] | 367 | if (big_pack_threshold) { |
Nguyễn Thái Ngọc Duy | 55dfe13 | 2018-04-15 17:36:15 +0200 | [diff] [blame] | 368 | find_base_packs(&keep_pack, big_pack_threshold); |
Nguyễn Thái Ngọc Duy | 8fc6776 | 2018-04-15 17:36:16 +0200 | [diff] [blame] | 369 | if (keep_pack.nr >= gc_auto_pack_limit) { |
| 370 | big_pack_threshold = 0; |
| 371 | string_list_clear(&keep_pack, 0); |
| 372 | find_base_packs(&keep_pack, 0); |
| 373 | } |
Nguyễn Thái Ngọc Duy | 9806f5a | 2018-04-15 17:36:17 +0200 | [diff] [blame] | 374 | } else { |
| 375 | struct packed_git *p = find_base_packs(&keep_pack, 0); |
| 376 | uint64_t mem_have, mem_want; |
| 377 | |
| 378 | mem_have = total_ram(); |
| 379 | mem_want = estimate_repack_memory(p); |
| 380 | |
| 381 | /* |
| 382 | * Only allow 1/2 of memory for pack-objects, leave |
| 383 | * the rest for the OS and other processes in the |
| 384 | * system. |
| 385 | */ |
| 386 | if (!mem_have || mem_want < mem_have / 2) |
| 387 | string_list_clear(&keep_pack, 0); |
Nguyễn Thái Ngọc Duy | 8fc6776 | 2018-04-15 17:36:16 +0200 | [diff] [blame] | 388 | } |
Nguyễn Thái Ngọc Duy | 55dfe13 | 2018-04-15 17:36:15 +0200 | [diff] [blame] | 389 | |
| 390 | add_repack_all_option(&keep_pack); |
| 391 | string_list_clear(&keep_pack, 0); |
| 392 | } else if (too_many_loose_objects()) |
David Turner | bdf56de | 2016-12-28 17:45:41 -0500 | [diff] [blame] | 393 | add_repack_incremental_option(); |
| 394 | else |
Junio C Hamano | 1781550 | 2007-09-17 00:55:13 -0700 | [diff] [blame] | 395 | return 0; |
Miklos Vajna | bde3054 | 2008-04-02 21:34:38 +0200 | [diff] [blame] | 396 | |
Benoit Pierre | 15048f8 | 2014-03-18 11:00:53 +0100 | [diff] [blame] | 397 | if (run_hook_le(NULL, "pre-auto-gc", NULL)) |
Miklos Vajna | bde3054 | 2008-04-02 21:34:38 +0200 | [diff] [blame] | 398 | return 0; |
Junio C Hamano | 95143f9 | 2007-09-17 00:48:39 -0700 | [diff] [blame] | 399 | return 1; |
Junio C Hamano | a087cc9 | 2007-09-17 00:44:17 -0700 | [diff] [blame] | 400 | } |
| 401 | |
Nguyễn Thái Ngọc Duy | 64a99eb | 2013-08-08 18:05:38 +0700 | [diff] [blame] | 402 | /* return NULL on success, else hostname running the gc */ |
| 403 | static const char *lock_repo_for_gc(int force, pid_t* ret_pid) |
| 404 | { |
Martin Ågren | b227586 | 2018-05-09 22:55:38 +0200 | [diff] [blame] | 405 | struct lock_file lock = LOCK_INIT; |
René Scharfe | da25bdb | 2017-04-18 17:57:42 -0400 | [diff] [blame] | 406 | char my_host[HOST_NAME_MAX + 1]; |
Nguyễn Thái Ngọc Duy | 64a99eb | 2013-08-08 18:05:38 +0700 | [diff] [blame] | 407 | struct strbuf sb = STRBUF_INIT; |
| 408 | struct stat st; |
| 409 | uintmax_t pid; |
| 410 | FILE *fp; |
Elia Pinto | 4f1c0b2 | 2014-01-29 08:59:37 -0800 | [diff] [blame] | 411 | int fd; |
Michael Haggerty | 00539ce | 2015-08-10 11:47:48 +0200 | [diff] [blame] | 412 | char *pidfile_path; |
Nguyễn Thái Ngọc Duy | 64a99eb | 2013-08-08 18:05:38 +0700 | [diff] [blame] | 413 | |
Jeff King | 076aa2c | 2017-09-05 08:15:08 -0400 | [diff] [blame] | 414 | if (is_tempfile_active(pidfile)) |
Jonathan Nieder | 4c5baf0 | 2013-10-16 16:11:46 -0700 | [diff] [blame] | 415 | /* already locked */ |
| 416 | return NULL; |
| 417 | |
David Turner | 5781a9a | 2017-04-18 17:57:43 -0400 | [diff] [blame] | 418 | if (xgethostname(my_host, sizeof(my_host))) |
Jeff King | 5096d49 | 2015-09-24 17:06:08 -0400 | [diff] [blame] | 419 | xsnprintf(my_host, sizeof(my_host), "unknown"); |
Nguyễn Thái Ngọc Duy | 64a99eb | 2013-08-08 18:05:38 +0700 | [diff] [blame] | 420 | |
Michael Haggerty | 00539ce | 2015-08-10 11:47:48 +0200 | [diff] [blame] | 421 | pidfile_path = git_pathdup("gc.pid"); |
| 422 | fd = hold_lock_file_for_update(&lock, pidfile_path, |
Nguyễn Thái Ngọc Duy | 64a99eb | 2013-08-08 18:05:38 +0700 | [diff] [blame] | 423 | LOCK_DIE_ON_ERROR); |
| 424 | if (!force) { |
René Scharfe | da25bdb | 2017-04-18 17:57:42 -0400 | [diff] [blame] | 425 | static char locking_host[HOST_NAME_MAX + 1]; |
| 426 | static char *scan_fmt; |
Elia Pinto | 4f1c0b2 | 2014-01-29 08:59:37 -0800 | [diff] [blame] | 427 | int should_exit; |
René Scharfe | da25bdb | 2017-04-18 17:57:42 -0400 | [diff] [blame] | 428 | |
| 429 | if (!scan_fmt) |
Junio C Hamano | afe2fab | 2017-09-17 12:16:55 +0900 | [diff] [blame] | 430 | scan_fmt = xstrfmt("%s %%%ds", "%"SCNuMAX, HOST_NAME_MAX); |
Michael Haggerty | 00539ce | 2015-08-10 11:47:48 +0200 | [diff] [blame] | 431 | fp = fopen(pidfile_path, "r"); |
Nguyễn Thái Ngọc Duy | 64a99eb | 2013-08-08 18:05:38 +0700 | [diff] [blame] | 432 | memset(locking_host, 0, sizeof(locking_host)); |
| 433 | should_exit = |
| 434 | fp != NULL && |
| 435 | !fstat(fileno(fp), &st) && |
| 436 | /* |
| 437 | * 12 hour limit is very generous as gc should |
| 438 | * never take that long. On the other hand we |
| 439 | * don't really need a strict limit here, |
| 440 | * running gc --auto one day late is not a big |
| 441 | * problem. --force can be used in manual gc |
| 442 | * after the user verifies that no gc is |
| 443 | * running. |
| 444 | */ |
| 445 | time(NULL) - st.st_mtime <= 12 * 3600 && |
René Scharfe | da25bdb | 2017-04-18 17:57:42 -0400 | [diff] [blame] | 446 | fscanf(fp, scan_fmt, &pid, locking_host) == 2 && |
Nguyễn Thái Ngọc Duy | 64a99eb | 2013-08-08 18:05:38 +0700 | [diff] [blame] | 447 | /* be gentle to concurrent "gc" on remote hosts */ |
Kyle J. McKay | ed7eda8 | 2013-12-31 04:07:39 -0800 | [diff] [blame] | 448 | (strcmp(locking_host, my_host) || !kill(pid, 0) || errno == EPERM); |
Nguyễn Thái Ngọc Duy | 64a99eb | 2013-08-08 18:05:38 +0700 | [diff] [blame] | 449 | if (fp != NULL) |
| 450 | fclose(fp); |
| 451 | if (should_exit) { |
| 452 | if (fd >= 0) |
| 453 | rollback_lock_file(&lock); |
| 454 | *ret_pid = pid; |
Michael Haggerty | 00539ce | 2015-08-10 11:47:48 +0200 | [diff] [blame] | 455 | free(pidfile_path); |
Nguyễn Thái Ngọc Duy | 64a99eb | 2013-08-08 18:05:38 +0700 | [diff] [blame] | 456 | return locking_host; |
| 457 | } |
| 458 | } |
| 459 | |
| 460 | strbuf_addf(&sb, "%"PRIuMAX" %s", |
| 461 | (uintmax_t) getpid(), my_host); |
| 462 | write_in_full(fd, sb.buf, sb.len); |
| 463 | strbuf_release(&sb); |
| 464 | commit_lock_file(&lock); |
Jeff King | 076aa2c | 2017-09-05 08:15:08 -0400 | [diff] [blame] | 465 | pidfile = register_tempfile(pidfile_path); |
Michael Haggerty | ebebeae | 2015-08-10 11:47:49 +0200 | [diff] [blame] | 466 | free(pidfile_path); |
Nguyễn Thái Ngọc Duy | 64a99eb | 2013-08-08 18:05:38 +0700 | [diff] [blame] | 467 | return NULL; |
| 468 | } |
| 469 | |
Jonathan Nieder | 3029970 | 2018-07-16 23:57:40 -0700 | [diff] [blame] | 470 | /* |
| 471 | * Returns 0 if there was no previous error and gc can proceed, 1 if |
| 472 | * gc should not proceed due to an error in the last run. Prints a |
Elijah Newren | 15beaaa | 2019-11-05 17:07:23 +0000 | [diff] [blame] | 473 | * message and returns -1 if an error occurred while reading gc.log |
Jonathan Nieder | 3029970 | 2018-07-16 23:57:40 -0700 | [diff] [blame] | 474 | */ |
| 475 | static int report_last_gc_error(void) |
Nguyễn Thái Ngọc Duy | 329e6e8 | 2015-09-19 12:13:23 +0700 | [diff] [blame] | 476 | { |
| 477 | struct strbuf sb = STRBUF_INIT; |
Jonathan Nieder | 3029970 | 2018-07-16 23:57:40 -0700 | [diff] [blame] | 478 | int ret = 0; |
Jonathan Nieder | 3c426ec | 2018-07-16 23:53:21 -0700 | [diff] [blame] | 479 | ssize_t len; |
David Turner | a831c06 | 2017-02-10 16:28:22 -0500 | [diff] [blame] | 480 | struct stat st; |
| 481 | char *gc_log_path = git_pathdup("gc.log"); |
Nguyễn Thái Ngọc Duy | 329e6e8 | 2015-09-19 12:13:23 +0700 | [diff] [blame] | 482 | |
David Turner | a831c06 | 2017-02-10 16:28:22 -0500 | [diff] [blame] | 483 | if (stat(gc_log_path, &st)) { |
| 484 | if (errno == ENOENT) |
| 485 | goto done; |
| 486 | |
Jonathan Nieder | 3029970 | 2018-07-16 23:57:40 -0700 | [diff] [blame] | 487 | ret = error_errno(_("cannot stat '%s'"), gc_log_path); |
| 488 | goto done; |
David Turner | a831c06 | 2017-02-10 16:28:22 -0500 | [diff] [blame] | 489 | } |
| 490 | |
| 491 | if (st.st_mtime < gc_log_expire_time) |
| 492 | goto done; |
| 493 | |
Jonathan Nieder | 3c426ec | 2018-07-16 23:53:21 -0700 | [diff] [blame] | 494 | len = strbuf_read_file(&sb, gc_log_path, 0); |
| 495 | if (len < 0) |
Jonathan Nieder | 3029970 | 2018-07-16 23:57:40 -0700 | [diff] [blame] | 496 | ret = error_errno(_("cannot read '%s'"), gc_log_path); |
| 497 | else if (len > 0) { |
| 498 | /* |
| 499 | * A previous gc failed. Report the error, and don't |
| 500 | * bother with an automatic gc run since it is likely |
| 501 | * to fail in the same way. |
| 502 | */ |
| 503 | warning(_("The last gc run reported the following. " |
Nguyễn Thái Ngọc Duy | 329e6e8 | 2015-09-19 12:13:23 +0700 | [diff] [blame] | 504 | "Please correct the root cause\n" |
Ævar Arnfjörð Bjarmason | b45c172 | 2021-08-31 16:37:29 +0200 | [diff] [blame] | 505 | "and remove %s\n" |
Nguyễn Thái Ngọc Duy | 329e6e8 | 2015-09-19 12:13:23 +0700 | [diff] [blame] | 506 | "Automatic cleanup will not be performed " |
| 507 | "until the file is removed.\n\n" |
| 508 | "%s"), |
David Turner | a831c06 | 2017-02-10 16:28:22 -0500 | [diff] [blame] | 509 | gc_log_path, sb.buf); |
Jonathan Nieder | 3029970 | 2018-07-16 23:57:40 -0700 | [diff] [blame] | 510 | ret = 1; |
| 511 | } |
Nguyễn Thái Ngọc Duy | 329e6e8 | 2015-09-19 12:13:23 +0700 | [diff] [blame] | 512 | strbuf_release(&sb); |
David Turner | a831c06 | 2017-02-10 16:28:22 -0500 | [diff] [blame] | 513 | done: |
| 514 | free(gc_log_path); |
Jonathan Nieder | 3029970 | 2018-07-16 23:57:40 -0700 | [diff] [blame] | 515 | return ret; |
Nguyễn Thái Ngọc Duy | 329e6e8 | 2015-09-19 12:13:23 +0700 | [diff] [blame] | 516 | } |
| 517 | |
Jonathan Nieder | fec2ed2 | 2018-07-16 23:54:16 -0700 | [diff] [blame] | 518 | static void gc_before_repack(void) |
Nguyễn Thái Ngọc Duy | 62aad18 | 2014-05-25 07:38:29 +0700 | [diff] [blame] | 519 | { |
Ævar Arnfjörð Bjarmason | cd8eb3a | 2019-03-15 16:59:54 +0100 | [diff] [blame] | 520 | /* |
| 521 | * We may be called twice, as both the pre- and |
| 522 | * post-daemonized phases will call us, but running these |
| 523 | * commands more than once is pointless and wasteful. |
| 524 | */ |
| 525 | static int done = 0; |
| 526 | if (done++) |
| 527 | return; |
| 528 | |
Derrick Stolee | 41abfe1 | 2021-02-09 13:42:28 +0000 | [diff] [blame] | 529 | if (pack_refs && maintenance_task_pack_refs(NULL)) |
| 530 | die(FAILED_RUN, "pack-refs"); |
Nguyễn Thái Ngọc Duy | 62aad18 | 2014-05-25 07:38:29 +0700 | [diff] [blame] | 531 | |
Jeff King | d70a9eb | 2020-07-28 20:37:20 -0400 | [diff] [blame] | 532 | if (prune_reflogs && run_command_v_opt(reflog.v, RUN_GIT_CMD)) |
| 533 | die(FAILED_RUN, reflog.v[0]); |
Nguyễn Thái Ngọc Duy | 62aad18 | 2014-05-25 07:38:29 +0700 | [diff] [blame] | 534 | } |
| 535 | |
James Bowes | 6757ada | 2007-03-13 21:58:22 -0400 | [diff] [blame] | 536 | int cmd_gc(int argc, const char **argv, const char *prefix) |
| 537 | { |
James Bowes | 44c637c | 2007-11-01 21:02:27 -0400 | [diff] [blame] | 538 | int aggressive = 0; |
Junio C Hamano | 2c3c439 | 2007-09-05 13:01:37 -0700 | [diff] [blame] | 539 | int auto_gc = 0; |
Frank Lichtenheld | a0c14cb | 2008-02-29 22:53:39 +0100 | [diff] [blame] | 540 | int quiet = 0; |
Nguyễn Thái Ngọc Duy | 64a99eb | 2013-08-08 18:05:38 +0700 | [diff] [blame] | 541 | int force = 0; |
| 542 | const char *name; |
| 543 | pid_t pid; |
Nguyễn Thái Ngọc Duy | 329e6e8 | 2015-09-19 12:13:23 +0700 | [diff] [blame] | 544 | int daemonized = 0; |
Ævar Arnfjörð Bjarmason | 793c146 | 2020-11-20 12:55:22 +0100 | [diff] [blame] | 545 | int keep_largest_pack = -1; |
Junio C Hamano | 8ab5aa4 | 2018-04-21 12:13:13 +0900 | [diff] [blame] | 546 | timestamp_t dummy; |
James Bowes | 6757ada | 2007-03-13 21:58:22 -0400 | [diff] [blame] | 547 | |
James Bowes | 44c637c | 2007-11-01 21:02:27 -0400 | [diff] [blame] | 548 | struct option builtin_gc_options[] = { |
Nguyễn Thái Ngọc Duy | 6705c16 | 2012-08-20 19:32:14 +0700 | [diff] [blame] | 549 | OPT__QUIET(&quiet, N_("suppress progress reporting")), |
| 550 | { OPTION_STRING, 0, "prune", &prune_expire, N_("date"), |
| 551 | N_("prune unreferenced objects"), |
Johannes Schindelin | 58e9d9d | 2009-02-14 23:10:10 +0100 | [diff] [blame] | 552 | PARSE_OPT_OPTARG, NULL, (intptr_t)prune_expire }, |
Stefan Beller | d5d09d4 | 2013-08-03 13:51:19 +0200 | [diff] [blame] | 553 | OPT_BOOL(0, "aggressive", &aggressive, N_("be more thorough (increased runtime)")), |
Nguyễn Thái Ngọc Duy | 7e1eeaa | 2018-02-09 18:01:58 +0700 | [diff] [blame] | 554 | OPT_BOOL_F(0, "auto", &auto_gc, N_("enable auto-gc mode"), |
| 555 | PARSE_OPT_NOCOMPLETE), |
| 556 | OPT_BOOL_F(0, "force", &force, |
| 557 | N_("force running gc even if there may be another gc running"), |
| 558 | PARSE_OPT_NOCOMPLETE), |
Ævar Arnfjörð Bjarmason | 793c146 | 2020-11-20 12:55:22 +0100 | [diff] [blame] | 559 | OPT_BOOL(0, "keep-largest-pack", &keep_largest_pack, |
Nguyễn Thái Ngọc Duy | ae4e89e | 2018-04-15 17:36:14 +0200 | [diff] [blame] | 560 | N_("repack all other packs except the largest pack")), |
James Bowes | 44c637c | 2007-11-01 21:02:27 -0400 | [diff] [blame] | 561 | OPT_END() |
| 562 | }; |
| 563 | |
Nguyễn Thái Ngọc Duy | 0c8151b | 2010-10-22 01:47:19 -0500 | [diff] [blame] | 564 | if (argc == 2 && !strcmp(argv[1], "-h")) |
| 565 | usage_with_options(builtin_gc_usage, builtin_gc_options); |
| 566 | |
Jeff King | 22f9b7f | 2020-07-28 16:24:27 -0400 | [diff] [blame] | 567 | strvec_pushl(&reflog, "reflog", "expire", "--all", NULL); |
| 568 | strvec_pushl(&repack, "repack", "-d", "-l", NULL); |
| 569 | strvec_pushl(&prune, "prune", "--expire", NULL); |
| 570 | strvec_pushl(&prune_worktrees, "worktree", "prune", "--expire", NULL); |
| 571 | strvec_pushl(&rerere, "rerere", "gc", NULL); |
Jeff King | 234587f | 2012-04-18 14:10:19 -0700 | [diff] [blame] | 572 | |
David Turner | a831c06 | 2017-02-10 16:28:22 -0500 | [diff] [blame] | 573 | /* default expiry time, overwritten in gc_config */ |
Tanay Abhra | 5801d3b | 2014-08-07 09:21:22 -0700 | [diff] [blame] | 574 | gc_config(); |
David Turner | a831c06 | 2017-02-10 16:28:22 -0500 | [diff] [blame] | 575 | if (parse_expiry_date(gc_log_expire, &gc_log_expire_time)) |
Junio C Hamano | 96913c9 | 2018-04-23 22:36:14 +0900 | [diff] [blame] | 576 | die(_("failed to parse gc.logexpiry value %s"), gc_log_expire); |
James Bowes | 6757ada | 2007-03-13 21:58:22 -0400 | [diff] [blame] | 577 | |
| 578 | if (pack_refs < 0) |
| 579 | pack_refs = !is_bare_repository(); |
| 580 | |
Stephen Boyd | 3778292 | 2009-05-23 11:53:12 -0700 | [diff] [blame] | 581 | argc = parse_options(argc, argv, prefix, builtin_gc_options, |
| 582 | builtin_gc_usage, 0); |
James Bowes | 44c637c | 2007-11-01 21:02:27 -0400 | [diff] [blame] | 583 | if (argc > 0) |
| 584 | usage_with_options(builtin_gc_usage, builtin_gc_options); |
| 585 | |
Junio C Hamano | 8ab5aa4 | 2018-04-21 12:13:13 +0900 | [diff] [blame] | 586 | if (prune_expire && parse_expiry_date(prune_expire, &dummy)) |
| 587 | die(_("failed to parse prune expiry value %s"), prune_expire); |
| 588 | |
James Bowes | 44c637c | 2007-11-01 21:02:27 -0400 | [diff] [blame] | 589 | if (aggressive) { |
Jeff King | 22f9b7f | 2020-07-28 16:24:27 -0400 | [diff] [blame] | 590 | strvec_push(&repack, "-f"); |
Nguyễn Thái Ngọc Duy | 125f814 | 2014-03-16 20:35:03 +0700 | [diff] [blame] | 591 | if (aggressive_depth > 0) |
Jeff King | 22f9b7f | 2020-07-28 16:24:27 -0400 | [diff] [blame] | 592 | strvec_pushf(&repack, "--depth=%d", aggressive_depth); |
Jeff King | 234587f | 2012-04-18 14:10:19 -0700 | [diff] [blame] | 593 | if (aggressive_window > 0) |
Jeff King | 22f9b7f | 2020-07-28 16:24:27 -0400 | [diff] [blame] | 594 | strvec_pushf(&repack, "--window=%d", aggressive_window); |
James Bowes | 6757ada | 2007-03-13 21:58:22 -0400 | [diff] [blame] | 595 | } |
Frank Lichtenheld | a0c14cb | 2008-02-29 22:53:39 +0100 | [diff] [blame] | 596 | if (quiet) |
Jeff King | 22f9b7f | 2020-07-28 16:24:27 -0400 | [diff] [blame] | 597 | strvec_push(&repack, "-q"); |
James Bowes | 6757ada | 2007-03-13 21:58:22 -0400 | [diff] [blame] | 598 | |
Junio C Hamano | 2c3c439 | 2007-09-05 13:01:37 -0700 | [diff] [blame] | 599 | if (auto_gc) { |
| 600 | /* |
| 601 | * Auto-gc should be least intrusive as possible. |
| 602 | */ |
Junio C Hamano | 2c3c439 | 2007-09-05 13:01:37 -0700 | [diff] [blame] | 603 | if (!need_to_gc()) |
| 604 | return 0; |
Nguyễn Thái Ngọc Duy | 9f673f9 | 2014-02-08 14:08:52 +0700 | [diff] [blame] | 605 | if (!quiet) { |
| 606 | if (detach_auto) |
| 607 | fprintf(stderr, _("Auto packing the repository in background for optimum performance.\n")); |
| 608 | else |
| 609 | fprintf(stderr, _("Auto packing the repository for optimum performance.\n")); |
| 610 | fprintf(stderr, _("See \"git help gc\" for manual housekeeping.\n")); |
| 611 | } |
Nguyễn Thái Ngọc Duy | 62aad18 | 2014-05-25 07:38:29 +0700 | [diff] [blame] | 612 | if (detach_auto) { |
Jonathan Nieder | 3029970 | 2018-07-16 23:57:40 -0700 | [diff] [blame] | 613 | int ret = report_last_gc_error(); |
| 614 | if (ret < 0) |
Elijah Newren | 15beaaa | 2019-11-05 17:07:23 +0000 | [diff] [blame] | 615 | /* an I/O error occurred, already reported */ |
Jonathan Nieder | 3029970 | 2018-07-16 23:57:40 -0700 | [diff] [blame] | 616 | exit(128); |
| 617 | if (ret == 1) |
| 618 | /* Last gc --auto failed. Skip this one. */ |
| 619 | return 0; |
Nguyễn Thái Ngọc Duy | 329e6e8 | 2015-09-19 12:13:23 +0700 | [diff] [blame] | 620 | |
Jeff King | c45af94 | 2017-07-11 05:06:35 -0400 | [diff] [blame] | 621 | if (lock_repo_for_gc(force, &pid)) |
| 622 | return 0; |
Jonathan Nieder | fec2ed2 | 2018-07-16 23:54:16 -0700 | [diff] [blame] | 623 | gc_before_repack(); /* dies on failure */ |
Jeff King | c45af94 | 2017-07-11 05:06:35 -0400 | [diff] [blame] | 624 | delete_tempfile(&pidfile); |
| 625 | |
Nguyễn Thái Ngọc Duy | 9f673f9 | 2014-02-08 14:08:52 +0700 | [diff] [blame] | 626 | /* |
| 627 | * failure to daemonize is ok, we'll continue |
| 628 | * in foreground |
| 629 | */ |
Nguyễn Thái Ngọc Duy | 329e6e8 | 2015-09-19 12:13:23 +0700 | [diff] [blame] | 630 | daemonized = !daemonize(); |
Nguyễn Thái Ngọc Duy | 62aad18 | 2014-05-25 07:38:29 +0700 | [diff] [blame] | 631 | } |
Nguyễn Thái Ngọc Duy | ae4e89e | 2018-04-15 17:36:14 +0200 | [diff] [blame] | 632 | } else { |
| 633 | struct string_list keep_pack = STRING_LIST_INIT_NODUP; |
| 634 | |
Ævar Arnfjörð Bjarmason | 793c146 | 2020-11-20 12:55:22 +0100 | [diff] [blame] | 635 | if (keep_largest_pack != -1) { |
| 636 | if (keep_largest_pack) |
Nguyễn Thái Ngọc Duy | 55dfe13 | 2018-04-15 17:36:15 +0200 | [diff] [blame] | 637 | find_base_packs(&keep_pack, 0); |
| 638 | } else if (big_pack_threshold) { |
| 639 | find_base_packs(&keep_pack, big_pack_threshold); |
Nguyễn Thái Ngọc Duy | ae4e89e | 2018-04-15 17:36:14 +0200 | [diff] [blame] | 640 | } |
| 641 | |
| 642 | add_repack_all_option(&keep_pack); |
| 643 | string_list_clear(&keep_pack, 0); |
| 644 | } |
Junio C Hamano | 2c3c439 | 2007-09-05 13:01:37 -0700 | [diff] [blame] | 645 | |
Nguyễn Thái Ngọc Duy | 64a99eb | 2013-08-08 18:05:38 +0700 | [diff] [blame] | 646 | name = lock_repo_for_gc(force, &pid); |
| 647 | if (name) { |
| 648 | if (auto_gc) |
| 649 | return 0; /* be quiet on --auto */ |
| 650 | die(_("gc is already running on machine '%s' pid %"PRIuMAX" (use --force if not)"), |
| 651 | name, (uintmax_t)pid); |
| 652 | } |
| 653 | |
Nguyễn Thái Ngọc Duy | 329e6e8 | 2015-09-19 12:13:23 +0700 | [diff] [blame] | 654 | if (daemonized) { |
| 655 | hold_lock_file_for_update(&log_lock, |
| 656 | git_path("gc.log"), |
| 657 | LOCK_DIE_ON_ERROR); |
Junio C Hamano | 076c827 | 2015-10-15 15:43:32 -0700 | [diff] [blame] | 658 | dup2(get_lock_file_fd(&log_lock), 2); |
Nguyễn Thái Ngọc Duy | 329e6e8 | 2015-09-19 12:13:23 +0700 | [diff] [blame] | 659 | sigchain_push_common(process_log_file_on_signal); |
| 660 | atexit(process_log_file_at_exit); |
| 661 | } |
| 662 | |
Jonathan Nieder | fec2ed2 | 2018-07-16 23:54:16 -0700 | [diff] [blame] | 663 | gc_before_repack(); |
James Bowes | 6757ada | 2007-03-13 21:58:22 -0400 | [diff] [blame] | 664 | |
Jeff King | 067fbd4 | 2015-06-23 06:54:11 -0400 | [diff] [blame] | 665 | if (!repository_format_precious_objects) { |
Derrick Stolee | 2d511cf | 2019-05-17 11:41:49 -0700 | [diff] [blame] | 666 | close_object_store(the_repository->objects); |
Jeff King | d70a9eb | 2020-07-28 20:37:20 -0400 | [diff] [blame] | 667 | if (run_command_v_opt(repack.v, RUN_GIT_CMD)) |
| 668 | die(FAILED_RUN, repack.v[0]); |
James Bowes | 6757ada | 2007-03-13 21:58:22 -0400 | [diff] [blame] | 669 | |
Jeff King | 067fbd4 | 2015-06-23 06:54:11 -0400 | [diff] [blame] | 670 | if (prune_expire) { |
Jeff King | 22f9b7f | 2020-07-28 16:24:27 -0400 | [diff] [blame] | 671 | strvec_push(&prune, prune_expire); |
Jeff King | 067fbd4 | 2015-06-23 06:54:11 -0400 | [diff] [blame] | 672 | if (quiet) |
Jeff King | 22f9b7f | 2020-07-28 16:24:27 -0400 | [diff] [blame] | 673 | strvec_push(&prune, "--no-progress"); |
Christian Couder | b14ed5a | 2019-06-25 15:40:31 +0200 | [diff] [blame] | 674 | if (has_promisor_remote()) |
Jeff King | 22f9b7f | 2020-07-28 16:24:27 -0400 | [diff] [blame] | 675 | strvec_push(&prune, |
Jeff King | f6d8942 | 2020-07-28 16:26:31 -0400 | [diff] [blame] | 676 | "--exclude-promisor-objects"); |
Jeff King | d70a9eb | 2020-07-28 20:37:20 -0400 | [diff] [blame] | 677 | if (run_command_v_opt(prune.v, RUN_GIT_CMD)) |
| 678 | die(FAILED_RUN, prune.v[0]); |
Jeff King | 067fbd4 | 2015-06-23 06:54:11 -0400 | [diff] [blame] | 679 | } |
Johannes Schindelin | 58e9d9d | 2009-02-14 23:10:10 +0100 | [diff] [blame] | 680 | } |
James Bowes | 6757ada | 2007-03-13 21:58:22 -0400 | [diff] [blame] | 681 | |
Nguyễn Thái Ngọc Duy | e3df33b | 2014-11-30 15:24:53 +0700 | [diff] [blame] | 682 | if (prune_worktrees_expire) { |
Jeff King | 22f9b7f | 2020-07-28 16:24:27 -0400 | [diff] [blame] | 683 | strvec_push(&prune_worktrees, prune_worktrees_expire); |
Jeff King | d70a9eb | 2020-07-28 20:37:20 -0400 | [diff] [blame] | 684 | if (run_command_v_opt(prune_worktrees.v, RUN_GIT_CMD)) |
| 685 | die(FAILED_RUN, prune_worktrees.v[0]); |
Nguyễn Thái Ngọc Duy | e3df33b | 2014-11-30 15:24:53 +0700 | [diff] [blame] | 686 | } |
| 687 | |
Jeff King | d70a9eb | 2020-07-28 20:37:20 -0400 | [diff] [blame] | 688 | if (run_command_v_opt(rerere.v, RUN_GIT_CMD)) |
| 689 | die(FAILED_RUN, rerere.v[0]); |
James Bowes | 6757ada | 2007-03-13 21:58:22 -0400 | [diff] [blame] | 690 | |
Doug Kelly | 478f34d | 2015-11-03 21:05:08 -0600 | [diff] [blame] | 691 | report_garbage = report_pack_garbage; |
Stefan Beller | a49d283 | 2018-03-23 18:45:21 +0100 | [diff] [blame] | 692 | reprepare_packed_git(the_repository); |
Johannes Schindelin | 5bdece0 | 2018-12-15 14:04:01 -0800 | [diff] [blame] | 693 | if (pack_garbage.nr > 0) { |
Derrick Stolee | 2d511cf | 2019-05-17 11:41:49 -0700 | [diff] [blame] | 694 | close_object_store(the_repository->objects); |
Doug Kelly | 478f34d | 2015-11-03 21:05:08 -0600 | [diff] [blame] | 695 | clean_pack_garbage(); |
Johannes Schindelin | 5bdece0 | 2018-12-15 14:04:01 -0800 | [diff] [blame] | 696 | } |
Doug Kelly | 478f34d | 2015-11-03 21:05:08 -0600 | [diff] [blame] | 697 | |
Derrick Stolee | 7211b9e | 2019-08-13 11:37:43 -0700 | [diff] [blame] | 698 | prepare_repo_settings(the_repository); |
| 699 | if (the_repository->settings.gc_write_commit_graph == 1) |
Taylor Blau | 0bd52e2 | 2020-02-03 21:51:50 -0800 | [diff] [blame] | 700 | write_commit_graph_reachable(the_repository->objects->odb, |
Junio C Hamano | f4f8dfe | 2019-09-09 12:26:36 -0700 | [diff] [blame] | 701 | !quiet && !daemonized ? COMMIT_GRAPH_WRITE_PROGRESS : 0, |
Derrick Stolee | 7211b9e | 2019-08-13 11:37:43 -0700 | [diff] [blame] | 702 | NULL); |
Derrick Stolee | d5d5d7b | 2018-06-27 09:24:46 -0400 | [diff] [blame] | 703 | |
Junio C Hamano | a087cc9 | 2007-09-17 00:44:17 -0700 | [diff] [blame] | 704 | if (auto_gc && too_many_loose_objects()) |
Ævar Arnfjörð Bjarmason | fea6128 | 2011-02-22 23:42:24 +0000 | [diff] [blame] | 705 | warning(_("There are too many unreachable loose objects; " |
| 706 | "run 'git prune' to remove them.")); |
Junio C Hamano | a087cc9 | 2007-09-17 00:44:17 -0700 | [diff] [blame] | 707 | |
David Turner | a831c06 | 2017-02-10 16:28:22 -0500 | [diff] [blame] | 708 | if (!daemonized) |
| 709 | unlink(git_path("gc.log")); |
| 710 | |
James Bowes | 6757ada | 2007-03-13 21:58:22 -0400 | [diff] [blame] | 711 | return 0; |
| 712 | } |
Derrick Stolee | 2057d75 | 2020-09-17 18:11:42 +0000 | [diff] [blame] | 713 | |
Derrick Stolee | b08ff1f | 2020-09-11 17:49:15 +0000 | [diff] [blame] | 714 | static const char *const builtin_maintenance_run_usage[] = { |
| 715 | N_("git maintenance run [--auto] [--[no-]quiet] [--task=<task>] [--schedule]"), |
Derrick Stolee | 2057d75 | 2020-09-17 18:11:42 +0000 | [diff] [blame] | 716 | NULL |
| 717 | }; |
| 718 | |
Derrick Stolee | b08ff1f | 2020-09-11 17:49:15 +0000 | [diff] [blame] | 719 | enum schedule_priority { |
| 720 | SCHEDULE_NONE = 0, |
| 721 | SCHEDULE_WEEKLY = 1, |
| 722 | SCHEDULE_DAILY = 2, |
| 723 | SCHEDULE_HOURLY = 3, |
| 724 | }; |
| 725 | |
| 726 | static enum schedule_priority parse_schedule(const char *value) |
| 727 | { |
| 728 | if (!value) |
| 729 | return SCHEDULE_NONE; |
| 730 | if (!strcasecmp(value, "hourly")) |
| 731 | return SCHEDULE_HOURLY; |
| 732 | if (!strcasecmp(value, "daily")) |
| 733 | return SCHEDULE_DAILY; |
| 734 | if (!strcasecmp(value, "weekly")) |
| 735 | return SCHEDULE_WEEKLY; |
| 736 | return SCHEDULE_NONE; |
| 737 | } |
| 738 | |
| 739 | static int maintenance_opt_schedule(const struct option *opt, const char *arg, |
| 740 | int unset) |
| 741 | { |
| 742 | enum schedule_priority *priority = opt->value; |
| 743 | |
| 744 | if (unset) |
| 745 | die(_("--no-schedule is not allowed")); |
| 746 | |
| 747 | *priority = parse_schedule(arg); |
| 748 | |
| 749 | if (!*priority) |
| 750 | die(_("unrecognized --schedule argument '%s'"), arg); |
| 751 | |
| 752 | return 0; |
| 753 | } |
| 754 | |
Derrick Stolee | 2057d75 | 2020-09-17 18:11:42 +0000 | [diff] [blame] | 755 | struct maintenance_run_opts { |
| 756 | int auto_flag; |
Derrick Stolee | 3ddaad0 | 2020-09-17 18:11:43 +0000 | [diff] [blame] | 757 | int quiet; |
Derrick Stolee | b08ff1f | 2020-09-11 17:49:15 +0000 | [diff] [blame] | 758 | enum schedule_priority schedule; |
Derrick Stolee | 2057d75 | 2020-09-17 18:11:42 +0000 | [diff] [blame] | 759 | }; |
| 760 | |
Derrick Stolee | 4ddc79b | 2020-09-17 18:11:51 +0000 | [diff] [blame] | 761 | /* Remember to update object flag allocation in object.h */ |
| 762 | #define SEEN (1u<<0) |
| 763 | |
| 764 | struct cg_auto_data { |
| 765 | int num_not_in_graph; |
| 766 | int limit; |
| 767 | }; |
| 768 | |
| 769 | static int dfs_on_ref(const char *refname, |
| 770 | const struct object_id *oid, int flags, |
| 771 | void *cb_data) |
| 772 | { |
| 773 | struct cg_auto_data *data = (struct cg_auto_data *)cb_data; |
| 774 | int result = 0; |
| 775 | struct object_id peeled; |
| 776 | struct commit_list *stack = NULL; |
| 777 | struct commit *commit; |
| 778 | |
Jeff King | 36a3179 | 2021-01-20 14:44:43 -0500 | [diff] [blame] | 779 | if (!peel_iterated_oid(oid, &peeled)) |
Derrick Stolee | 4ddc79b | 2020-09-17 18:11:51 +0000 | [diff] [blame] | 780 | oid = &peeled; |
| 781 | if (oid_object_info(the_repository, oid, NULL) != OBJ_COMMIT) |
| 782 | return 0; |
| 783 | |
| 784 | commit = lookup_commit(the_repository, oid); |
| 785 | if (!commit) |
| 786 | return 0; |
Derrick Stolee | 8f80180 | 2020-10-08 00:50:39 +0000 | [diff] [blame] | 787 | if (parse_commit(commit) || |
| 788 | commit_graph_position(commit) != COMMIT_NOT_FROM_GRAPH) |
Derrick Stolee | 4ddc79b | 2020-09-17 18:11:51 +0000 | [diff] [blame] | 789 | return 0; |
| 790 | |
Derrick Stolee | 8f80180 | 2020-10-08 00:50:39 +0000 | [diff] [blame] | 791 | data->num_not_in_graph++; |
| 792 | |
| 793 | if (data->num_not_in_graph >= data->limit) |
| 794 | return 1; |
| 795 | |
Derrick Stolee | 4ddc79b | 2020-09-17 18:11:51 +0000 | [diff] [blame] | 796 | commit_list_append(commit, &stack); |
| 797 | |
| 798 | while (!result && stack) { |
| 799 | struct commit_list *parent; |
| 800 | |
| 801 | commit = pop_commit(&stack); |
| 802 | |
| 803 | for (parent = commit->parents; parent; parent = parent->next) { |
| 804 | if (parse_commit(parent->item) || |
| 805 | commit_graph_position(parent->item) != COMMIT_NOT_FROM_GRAPH || |
| 806 | parent->item->object.flags & SEEN) |
| 807 | continue; |
| 808 | |
| 809 | parent->item->object.flags |= SEEN; |
| 810 | data->num_not_in_graph++; |
| 811 | |
| 812 | if (data->num_not_in_graph >= data->limit) { |
| 813 | result = 1; |
| 814 | break; |
| 815 | } |
| 816 | |
| 817 | commit_list_append(parent->item, &stack); |
| 818 | } |
| 819 | } |
| 820 | |
| 821 | free_commit_list(stack); |
| 822 | return result; |
| 823 | } |
| 824 | |
| 825 | static int should_write_commit_graph(void) |
| 826 | { |
| 827 | int result; |
| 828 | struct cg_auto_data data; |
| 829 | |
| 830 | data.num_not_in_graph = 0; |
| 831 | data.limit = 100; |
| 832 | git_config_get_int("maintenance.commit-graph.auto", |
| 833 | &data.limit); |
| 834 | |
| 835 | if (!data.limit) |
| 836 | return 0; |
| 837 | if (data.limit < 0) |
| 838 | return 1; |
| 839 | |
| 840 | result = for_each_ref(dfs_on_ref, &data); |
| 841 | |
René Scharfe | cd88884 | 2020-10-31 13:46:08 +0100 | [diff] [blame] | 842 | repo_clear_commit_marks(the_repository, SEEN); |
Derrick Stolee | 4ddc79b | 2020-09-17 18:11:51 +0000 | [diff] [blame] | 843 | |
| 844 | return result; |
| 845 | } |
| 846 | |
Derrick Stolee | 663b2b1 | 2020-09-17 18:11:46 +0000 | [diff] [blame] | 847 | static int run_write_commit_graph(struct maintenance_run_opts *opts) |
| 848 | { |
| 849 | struct child_process child = CHILD_PROCESS_INIT; |
| 850 | |
| 851 | child.git_cmd = 1; |
| 852 | strvec_pushl(&child.args, "commit-graph", "write", |
| 853 | "--split", "--reachable", NULL); |
| 854 | |
| 855 | if (opts->quiet) |
| 856 | strvec_push(&child.args, "--no-progress"); |
| 857 | |
| 858 | return !!run_command(&child); |
| 859 | } |
| 860 | |
| 861 | static int maintenance_task_commit_graph(struct maintenance_run_opts *opts) |
| 862 | { |
Derrick Stolee | d334107 | 2020-10-12 13:28:34 +0000 | [diff] [blame] | 863 | prepare_repo_settings(the_repository); |
| 864 | if (!the_repository->settings.core_commit_graph) |
| 865 | return 0; |
| 866 | |
Derrick Stolee | 663b2b1 | 2020-09-17 18:11:46 +0000 | [diff] [blame] | 867 | close_object_store(the_repository->objects); |
| 868 | if (run_write_commit_graph(opts)) { |
| 869 | error(_("failed to write commit-graph")); |
| 870 | return 1; |
| 871 | } |
| 872 | |
| 873 | return 0; |
| 874 | } |
| 875 | |
Derrick Stolee | a039a1f | 2021-04-06 18:47:46 +0000 | [diff] [blame] | 876 | static int fetch_remote(struct remote *remote, void *cbdata) |
Derrick Stolee | 28cb5e6 | 2020-09-25 12:33:31 +0000 | [diff] [blame] | 877 | { |
Derrick Stolee | a039a1f | 2021-04-06 18:47:46 +0000 | [diff] [blame] | 878 | struct maintenance_run_opts *opts = cbdata; |
Derrick Stolee | 28cb5e6 | 2020-09-25 12:33:31 +0000 | [diff] [blame] | 879 | struct child_process child = CHILD_PROCESS_INIT; |
| 880 | |
Derrick Stolee | 32f6788 | 2021-04-16 12:49:59 +0000 | [diff] [blame] | 881 | if (remote->skip_default_update) |
| 882 | return 0; |
| 883 | |
Derrick Stolee | 28cb5e6 | 2020-09-25 12:33:31 +0000 | [diff] [blame] | 884 | child.git_cmd = 1; |
Derrick Stolee | cfd781e | 2021-04-16 12:49:58 +0000 | [diff] [blame] | 885 | strvec_pushl(&child.args, "fetch", remote->name, |
| 886 | "--prefetch", "--prune", "--no-tags", |
Derrick Stolee | 28cb5e6 | 2020-09-25 12:33:31 +0000 | [diff] [blame] | 887 | "--no-write-fetch-head", "--recurse-submodules=no", |
Derrick Stolee | cfd781e | 2021-04-16 12:49:58 +0000 | [diff] [blame] | 888 | NULL); |
Derrick Stolee | 28cb5e6 | 2020-09-25 12:33:31 +0000 | [diff] [blame] | 889 | |
| 890 | if (opts->quiet) |
| 891 | strvec_push(&child.args, "--quiet"); |
| 892 | |
Derrick Stolee | 28cb5e6 | 2020-09-25 12:33:31 +0000 | [diff] [blame] | 893 | return !!run_command(&child); |
| 894 | } |
| 895 | |
Derrick Stolee | 28cb5e6 | 2020-09-25 12:33:31 +0000 | [diff] [blame] | 896 | static int maintenance_task_prefetch(struct maintenance_run_opts *opts) |
| 897 | { |
Derrick Stolee | 96eaffe | 2021-01-19 12:52:03 +0000 | [diff] [blame] | 898 | git_config_set_multivar_gently("log.excludedecoration", |
| 899 | "refs/prefetch/", |
| 900 | "refs/prefetch/", |
| 901 | CONFIG_FLAGS_FIXED_VALUE | |
| 902 | CONFIG_FLAGS_MULTI_REPLACE); |
| 903 | |
Derrick Stolee | a039a1f | 2021-04-06 18:47:46 +0000 | [diff] [blame] | 904 | if (for_each_remote(fetch_remote, opts)) { |
| 905 | error(_("failed to prefetch remotes")); |
| 906 | return 1; |
Derrick Stolee | 28cb5e6 | 2020-09-25 12:33:31 +0000 | [diff] [blame] | 907 | } |
| 908 | |
Derrick Stolee | a039a1f | 2021-04-06 18:47:46 +0000 | [diff] [blame] | 909 | return 0; |
Derrick Stolee | 28cb5e6 | 2020-09-25 12:33:31 +0000 | [diff] [blame] | 910 | } |
| 911 | |
Derrick Stolee | 2057d75 | 2020-09-17 18:11:42 +0000 | [diff] [blame] | 912 | static int maintenance_task_gc(struct maintenance_run_opts *opts) |
| 913 | { |
| 914 | struct child_process child = CHILD_PROCESS_INIT; |
| 915 | |
| 916 | child.git_cmd = 1; |
| 917 | strvec_push(&child.args, "gc"); |
| 918 | |
| 919 | if (opts->auto_flag) |
| 920 | strvec_push(&child.args, "--auto"); |
Derrick Stolee | 3ddaad0 | 2020-09-17 18:11:43 +0000 | [diff] [blame] | 921 | if (opts->quiet) |
| 922 | strvec_push(&child.args, "--quiet"); |
| 923 | else |
| 924 | strvec_push(&child.args, "--no-quiet"); |
Derrick Stolee | 2057d75 | 2020-09-17 18:11:42 +0000 | [diff] [blame] | 925 | |
| 926 | close_object_store(the_repository->objects); |
| 927 | return run_command(&child); |
| 928 | } |
| 929 | |
Derrick Stolee | 252cfb7 | 2020-09-25 12:33:32 +0000 | [diff] [blame] | 930 | static int prune_packed(struct maintenance_run_opts *opts) |
| 931 | { |
| 932 | struct child_process child = CHILD_PROCESS_INIT; |
| 933 | |
| 934 | child.git_cmd = 1; |
| 935 | strvec_push(&child.args, "prune-packed"); |
| 936 | |
| 937 | if (opts->quiet) |
| 938 | strvec_push(&child.args, "--quiet"); |
| 939 | |
| 940 | return !!run_command(&child); |
| 941 | } |
| 942 | |
| 943 | struct write_loose_object_data { |
| 944 | FILE *in; |
| 945 | int count; |
| 946 | int batch_size; |
| 947 | }; |
| 948 | |
Derrick Stolee | 3e220e6 | 2020-09-25 12:33:33 +0000 | [diff] [blame] | 949 | static int loose_object_auto_limit = 100; |
| 950 | |
| 951 | static int loose_object_count(const struct object_id *oid, |
| 952 | const char *path, |
| 953 | void *data) |
| 954 | { |
| 955 | int *count = (int*)data; |
| 956 | if (++(*count) >= loose_object_auto_limit) |
| 957 | return 1; |
| 958 | return 0; |
| 959 | } |
| 960 | |
| 961 | static int loose_object_auto_condition(void) |
| 962 | { |
| 963 | int count = 0; |
| 964 | |
| 965 | git_config_get_int("maintenance.loose-objects.auto", |
| 966 | &loose_object_auto_limit); |
| 967 | |
| 968 | if (!loose_object_auto_limit) |
| 969 | return 0; |
| 970 | if (loose_object_auto_limit < 0) |
| 971 | return 1; |
| 972 | |
| 973 | return for_each_loose_file_in_objdir(the_repository->objects->odb->path, |
| 974 | loose_object_count, |
| 975 | NULL, NULL, &count); |
| 976 | } |
| 977 | |
Derrick Stolee | 252cfb7 | 2020-09-25 12:33:32 +0000 | [diff] [blame] | 978 | static int bail_on_loose(const struct object_id *oid, |
| 979 | const char *path, |
| 980 | void *data) |
| 981 | { |
| 982 | return 1; |
| 983 | } |
| 984 | |
| 985 | static int write_loose_object_to_stdin(const struct object_id *oid, |
| 986 | const char *path, |
| 987 | void *data) |
| 988 | { |
| 989 | struct write_loose_object_data *d = (struct write_loose_object_data *)data; |
| 990 | |
| 991 | fprintf(d->in, "%s\n", oid_to_hex(oid)); |
| 992 | |
| 993 | return ++(d->count) > d->batch_size; |
| 994 | } |
| 995 | |
| 996 | static int pack_loose(struct maintenance_run_opts *opts) |
| 997 | { |
| 998 | struct repository *r = the_repository; |
| 999 | int result = 0; |
| 1000 | struct write_loose_object_data data; |
| 1001 | struct child_process pack_proc = CHILD_PROCESS_INIT; |
| 1002 | |
| 1003 | /* |
| 1004 | * Do not start pack-objects process |
| 1005 | * if there are no loose objects. |
| 1006 | */ |
| 1007 | if (!for_each_loose_file_in_objdir(r->objects->odb->path, |
| 1008 | bail_on_loose, |
| 1009 | NULL, NULL, NULL)) |
| 1010 | return 0; |
| 1011 | |
| 1012 | pack_proc.git_cmd = 1; |
| 1013 | |
| 1014 | strvec_push(&pack_proc.args, "pack-objects"); |
| 1015 | if (opts->quiet) |
| 1016 | strvec_push(&pack_proc.args, "--quiet"); |
| 1017 | strvec_pushf(&pack_proc.args, "%s/pack/loose", r->objects->odb->path); |
| 1018 | |
| 1019 | pack_proc.in = -1; |
| 1020 | |
| 1021 | if (start_command(&pack_proc)) { |
| 1022 | error(_("failed to start 'git pack-objects' process")); |
| 1023 | return 1; |
| 1024 | } |
| 1025 | |
| 1026 | data.in = xfdopen(pack_proc.in, "w"); |
| 1027 | data.count = 0; |
| 1028 | data.batch_size = 50000; |
| 1029 | |
| 1030 | for_each_loose_file_in_objdir(r->objects->odb->path, |
| 1031 | write_loose_object_to_stdin, |
| 1032 | NULL, |
| 1033 | NULL, |
| 1034 | &data); |
| 1035 | |
| 1036 | fclose(data.in); |
| 1037 | |
| 1038 | if (finish_command(&pack_proc)) { |
| 1039 | error(_("failed to finish 'git pack-objects' process")); |
| 1040 | result = 1; |
| 1041 | } |
| 1042 | |
| 1043 | return result; |
| 1044 | } |
| 1045 | |
| 1046 | static int maintenance_task_loose_objects(struct maintenance_run_opts *opts) |
| 1047 | { |
| 1048 | return prune_packed(opts) || pack_loose(opts); |
| 1049 | } |
| 1050 | |
Derrick Stolee | e841a79 | 2020-09-25 12:33:38 +0000 | [diff] [blame] | 1051 | static int incremental_repack_auto_condition(void) |
| 1052 | { |
| 1053 | struct packed_git *p; |
| 1054 | int enabled; |
| 1055 | int incremental_repack_auto_limit = 10; |
| 1056 | int count = 0; |
| 1057 | |
| 1058 | if (git_config_get_bool("core.multiPackIndex", &enabled) || |
| 1059 | !enabled) |
| 1060 | return 0; |
| 1061 | |
| 1062 | git_config_get_int("maintenance.incremental-repack.auto", |
| 1063 | &incremental_repack_auto_limit); |
| 1064 | |
| 1065 | if (!incremental_repack_auto_limit) |
| 1066 | return 0; |
| 1067 | if (incremental_repack_auto_limit < 0) |
| 1068 | return 1; |
| 1069 | |
| 1070 | for (p = get_packed_git(the_repository); |
| 1071 | count < incremental_repack_auto_limit && p; |
| 1072 | p = p->next) { |
| 1073 | if (!p->multi_pack_index) |
| 1074 | count++; |
| 1075 | } |
| 1076 | |
| 1077 | return count >= incremental_repack_auto_limit; |
| 1078 | } |
| 1079 | |
Derrick Stolee | 52fe41f | 2020-09-25 12:33:36 +0000 | [diff] [blame] | 1080 | static int multi_pack_index_write(struct maintenance_run_opts *opts) |
| 1081 | { |
| 1082 | struct child_process child = CHILD_PROCESS_INIT; |
| 1083 | |
| 1084 | child.git_cmd = 1; |
| 1085 | strvec_pushl(&child.args, "multi-pack-index", "write", NULL); |
| 1086 | |
| 1087 | if (opts->quiet) |
| 1088 | strvec_push(&child.args, "--no-progress"); |
| 1089 | |
| 1090 | if (run_command(&child)) |
| 1091 | return error(_("failed to write multi-pack-index")); |
| 1092 | |
| 1093 | return 0; |
| 1094 | } |
| 1095 | |
| 1096 | static int multi_pack_index_expire(struct maintenance_run_opts *opts) |
| 1097 | { |
| 1098 | struct child_process child = CHILD_PROCESS_INIT; |
| 1099 | |
| 1100 | child.git_cmd = 1; |
| 1101 | strvec_pushl(&child.args, "multi-pack-index", "expire", NULL); |
| 1102 | |
| 1103 | if (opts->quiet) |
| 1104 | strvec_push(&child.args, "--no-progress"); |
| 1105 | |
| 1106 | close_object_store(the_repository->objects); |
| 1107 | |
| 1108 | if (run_command(&child)) |
| 1109 | return error(_("'git multi-pack-index expire' failed")); |
| 1110 | |
| 1111 | return 0; |
| 1112 | } |
| 1113 | |
Derrick Stolee | a13e3d0 | 2020-09-25 12:33:37 +0000 | [diff] [blame] | 1114 | #define TWO_GIGABYTES (INT32_MAX) |
| 1115 | |
| 1116 | static off_t get_auto_pack_size(void) |
| 1117 | { |
| 1118 | /* |
| 1119 | * The "auto" value is special: we optimize for |
| 1120 | * one large pack-file (i.e. from a clone) and |
| 1121 | * expect the rest to be small and they can be |
| 1122 | * repacked quickly. |
| 1123 | * |
| 1124 | * The strategy we select here is to select a |
| 1125 | * size that is one more than the second largest |
| 1126 | * pack-file. This ensures that we will repack |
| 1127 | * at least two packs if there are three or more |
| 1128 | * packs. |
| 1129 | */ |
| 1130 | off_t max_size = 0; |
| 1131 | off_t second_largest_size = 0; |
| 1132 | off_t result_size; |
| 1133 | struct packed_git *p; |
| 1134 | struct repository *r = the_repository; |
| 1135 | |
| 1136 | reprepare_packed_git(r); |
| 1137 | for (p = get_all_packs(r); p; p = p->next) { |
| 1138 | if (p->pack_size > max_size) { |
| 1139 | second_largest_size = max_size; |
| 1140 | max_size = p->pack_size; |
| 1141 | } else if (p->pack_size > second_largest_size) |
| 1142 | second_largest_size = p->pack_size; |
| 1143 | } |
| 1144 | |
| 1145 | result_size = second_largest_size + 1; |
| 1146 | |
| 1147 | /* But limit ourselves to a batch size of 2g */ |
| 1148 | if (result_size > TWO_GIGABYTES) |
| 1149 | result_size = TWO_GIGABYTES; |
| 1150 | |
| 1151 | return result_size; |
| 1152 | } |
| 1153 | |
Derrick Stolee | 52fe41f | 2020-09-25 12:33:36 +0000 | [diff] [blame] | 1154 | static int multi_pack_index_repack(struct maintenance_run_opts *opts) |
| 1155 | { |
| 1156 | struct child_process child = CHILD_PROCESS_INIT; |
| 1157 | |
| 1158 | child.git_cmd = 1; |
| 1159 | strvec_pushl(&child.args, "multi-pack-index", "repack", NULL); |
| 1160 | |
| 1161 | if (opts->quiet) |
| 1162 | strvec_push(&child.args, "--no-progress"); |
| 1163 | |
Derrick Stolee | a13e3d0 | 2020-09-25 12:33:37 +0000 | [diff] [blame] | 1164 | strvec_pushf(&child.args, "--batch-size=%"PRIuMAX, |
| 1165 | (uintmax_t)get_auto_pack_size()); |
Derrick Stolee | 52fe41f | 2020-09-25 12:33:36 +0000 | [diff] [blame] | 1166 | |
| 1167 | close_object_store(the_repository->objects); |
| 1168 | |
| 1169 | if (run_command(&child)) |
| 1170 | return error(_("'git multi-pack-index repack' failed")); |
| 1171 | |
| 1172 | return 0; |
| 1173 | } |
| 1174 | |
| 1175 | static int maintenance_task_incremental_repack(struct maintenance_run_opts *opts) |
| 1176 | { |
| 1177 | prepare_repo_settings(the_repository); |
| 1178 | if (!the_repository->settings.core_multi_pack_index) { |
| 1179 | warning(_("skipping incremental-repack task because core.multiPackIndex is disabled")); |
| 1180 | return 0; |
| 1181 | } |
| 1182 | |
| 1183 | if (multi_pack_index_write(opts)) |
| 1184 | return 1; |
| 1185 | if (multi_pack_index_expire(opts)) |
| 1186 | return 1; |
| 1187 | if (multi_pack_index_repack(opts)) |
| 1188 | return 1; |
| 1189 | return 0; |
| 1190 | } |
| 1191 | |
Derrick Stolee | 3103e98 | 2020-09-17 18:11:45 +0000 | [diff] [blame] | 1192 | typedef int maintenance_task_fn(struct maintenance_run_opts *opts); |
| 1193 | |
Derrick Stolee | 916d062 | 2020-09-17 18:11:50 +0000 | [diff] [blame] | 1194 | /* |
| 1195 | * An auto condition function returns 1 if the task should run |
| 1196 | * and 0 if the task should NOT run. See needs_to_gc() for an |
| 1197 | * example. |
| 1198 | */ |
| 1199 | typedef int maintenance_auto_fn(void); |
| 1200 | |
Derrick Stolee | 3103e98 | 2020-09-17 18:11:45 +0000 | [diff] [blame] | 1201 | struct maintenance_task { |
| 1202 | const char *name; |
| 1203 | maintenance_task_fn *fn; |
Derrick Stolee | 916d062 | 2020-09-17 18:11:50 +0000 | [diff] [blame] | 1204 | maintenance_auto_fn *auto_condition; |
Derrick Stolee | 3103e98 | 2020-09-17 18:11:45 +0000 | [diff] [blame] | 1205 | unsigned enabled:1; |
Derrick Stolee | 090511b | 2020-09-17 18:11:47 +0000 | [diff] [blame] | 1206 | |
Derrick Stolee | b08ff1f | 2020-09-11 17:49:15 +0000 | [diff] [blame] | 1207 | enum schedule_priority schedule; |
| 1208 | |
Derrick Stolee | 090511b | 2020-09-17 18:11:47 +0000 | [diff] [blame] | 1209 | /* -1 if not selected. */ |
| 1210 | int selected_order; |
Derrick Stolee | 3103e98 | 2020-09-17 18:11:45 +0000 | [diff] [blame] | 1211 | }; |
| 1212 | |
| 1213 | enum maintenance_task_label { |
Derrick Stolee | 28cb5e6 | 2020-09-25 12:33:31 +0000 | [diff] [blame] | 1214 | TASK_PREFETCH, |
Derrick Stolee | 252cfb7 | 2020-09-25 12:33:32 +0000 | [diff] [blame] | 1215 | TASK_LOOSE_OBJECTS, |
Derrick Stolee | 52fe41f | 2020-09-25 12:33:36 +0000 | [diff] [blame] | 1216 | TASK_INCREMENTAL_REPACK, |
Derrick Stolee | 3103e98 | 2020-09-17 18:11:45 +0000 | [diff] [blame] | 1217 | TASK_GC, |
Derrick Stolee | 663b2b1 | 2020-09-17 18:11:46 +0000 | [diff] [blame] | 1218 | TASK_COMMIT_GRAPH, |
Derrick Stolee | 41abfe1 | 2021-02-09 13:42:28 +0000 | [diff] [blame] | 1219 | TASK_PACK_REFS, |
Derrick Stolee | 3103e98 | 2020-09-17 18:11:45 +0000 | [diff] [blame] | 1220 | |
| 1221 | /* Leave as final value */ |
| 1222 | TASK__COUNT |
| 1223 | }; |
| 1224 | |
| 1225 | static struct maintenance_task tasks[] = { |
Derrick Stolee | 28cb5e6 | 2020-09-25 12:33:31 +0000 | [diff] [blame] | 1226 | [TASK_PREFETCH] = { |
| 1227 | "prefetch", |
| 1228 | maintenance_task_prefetch, |
| 1229 | }, |
Derrick Stolee | 252cfb7 | 2020-09-25 12:33:32 +0000 | [diff] [blame] | 1230 | [TASK_LOOSE_OBJECTS] = { |
| 1231 | "loose-objects", |
| 1232 | maintenance_task_loose_objects, |
Derrick Stolee | 3e220e6 | 2020-09-25 12:33:33 +0000 | [diff] [blame] | 1233 | loose_object_auto_condition, |
Derrick Stolee | 252cfb7 | 2020-09-25 12:33:32 +0000 | [diff] [blame] | 1234 | }, |
Derrick Stolee | 52fe41f | 2020-09-25 12:33:36 +0000 | [diff] [blame] | 1235 | [TASK_INCREMENTAL_REPACK] = { |
| 1236 | "incremental-repack", |
| 1237 | maintenance_task_incremental_repack, |
Derrick Stolee | e841a79 | 2020-09-25 12:33:38 +0000 | [diff] [blame] | 1238 | incremental_repack_auto_condition, |
Derrick Stolee | 52fe41f | 2020-09-25 12:33:36 +0000 | [diff] [blame] | 1239 | }, |
Derrick Stolee | 3103e98 | 2020-09-17 18:11:45 +0000 | [diff] [blame] | 1240 | [TASK_GC] = { |
| 1241 | "gc", |
| 1242 | maintenance_task_gc, |
Derrick Stolee | 916d062 | 2020-09-17 18:11:50 +0000 | [diff] [blame] | 1243 | need_to_gc, |
Derrick Stolee | 3103e98 | 2020-09-17 18:11:45 +0000 | [diff] [blame] | 1244 | 1, |
| 1245 | }, |
Derrick Stolee | 663b2b1 | 2020-09-17 18:11:46 +0000 | [diff] [blame] | 1246 | [TASK_COMMIT_GRAPH] = { |
| 1247 | "commit-graph", |
| 1248 | maintenance_task_commit_graph, |
Derrick Stolee | 4ddc79b | 2020-09-17 18:11:51 +0000 | [diff] [blame] | 1249 | should_write_commit_graph, |
Derrick Stolee | 663b2b1 | 2020-09-17 18:11:46 +0000 | [diff] [blame] | 1250 | }, |
Derrick Stolee | 41abfe1 | 2021-02-09 13:42:28 +0000 | [diff] [blame] | 1251 | [TASK_PACK_REFS] = { |
| 1252 | "pack-refs", |
| 1253 | maintenance_task_pack_refs, |
| 1254 | NULL, |
| 1255 | }, |
Derrick Stolee | 3103e98 | 2020-09-17 18:11:45 +0000 | [diff] [blame] | 1256 | }; |
| 1257 | |
Derrick Stolee | 090511b | 2020-09-17 18:11:47 +0000 | [diff] [blame] | 1258 | static int compare_tasks_by_selection(const void *a_, const void *b_) |
| 1259 | { |
René Scharfe | a1c7479 | 2020-11-17 22:59:49 +0100 | [diff] [blame] | 1260 | const struct maintenance_task *a = a_; |
| 1261 | const struct maintenance_task *b = b_; |
Derrick Stolee | 090511b | 2020-09-17 18:11:47 +0000 | [diff] [blame] | 1262 | |
| 1263 | return b->selected_order - a->selected_order; |
| 1264 | } |
| 1265 | |
Derrick Stolee | 3103e98 | 2020-09-17 18:11:45 +0000 | [diff] [blame] | 1266 | static int maintenance_run_tasks(struct maintenance_run_opts *opts) |
| 1267 | { |
Derrick Stolee | 090511b | 2020-09-17 18:11:47 +0000 | [diff] [blame] | 1268 | int i, found_selected = 0; |
Derrick Stolee | 3103e98 | 2020-09-17 18:11:45 +0000 | [diff] [blame] | 1269 | int result = 0; |
Derrick Stolee | d7514f6 | 2020-09-17 18:11:48 +0000 | [diff] [blame] | 1270 | struct lock_file lk; |
| 1271 | struct repository *r = the_repository; |
| 1272 | char *lock_path = xstrfmt("%s/maintenance", r->objects->odb->path); |
| 1273 | |
| 1274 | if (hold_lock_file_for_update(&lk, lock_path, LOCK_NO_DEREF) < 0) { |
| 1275 | /* |
| 1276 | * Another maintenance command is running. |
| 1277 | * |
| 1278 | * If --auto was provided, then it is likely due to a |
| 1279 | * recursive process stack. Do not report an error in |
| 1280 | * that case. |
| 1281 | */ |
| 1282 | if (!opts->auto_flag && !opts->quiet) |
| 1283 | warning(_("lock file '%s' exists, skipping maintenance"), |
| 1284 | lock_path); |
| 1285 | free(lock_path); |
| 1286 | return 0; |
| 1287 | } |
| 1288 | free(lock_path); |
Derrick Stolee | 3103e98 | 2020-09-17 18:11:45 +0000 | [diff] [blame] | 1289 | |
Derrick Stolee | 090511b | 2020-09-17 18:11:47 +0000 | [diff] [blame] | 1290 | for (i = 0; !found_selected && i < TASK__COUNT; i++) |
| 1291 | found_selected = tasks[i].selected_order >= 0; |
| 1292 | |
| 1293 | if (found_selected) |
| 1294 | QSORT(tasks, TASK__COUNT, compare_tasks_by_selection); |
| 1295 | |
Derrick Stolee | 3103e98 | 2020-09-17 18:11:45 +0000 | [diff] [blame] | 1296 | for (i = 0; i < TASK__COUNT; i++) { |
Derrick Stolee | 090511b | 2020-09-17 18:11:47 +0000 | [diff] [blame] | 1297 | if (found_selected && tasks[i].selected_order < 0) |
| 1298 | continue; |
| 1299 | |
| 1300 | if (!found_selected && !tasks[i].enabled) |
Derrick Stolee | 3103e98 | 2020-09-17 18:11:45 +0000 | [diff] [blame] | 1301 | continue; |
| 1302 | |
Derrick Stolee | 916d062 | 2020-09-17 18:11:50 +0000 | [diff] [blame] | 1303 | if (opts->auto_flag && |
| 1304 | (!tasks[i].auto_condition || |
| 1305 | !tasks[i].auto_condition())) |
| 1306 | continue; |
| 1307 | |
Derrick Stolee | b08ff1f | 2020-09-11 17:49:15 +0000 | [diff] [blame] | 1308 | if (opts->schedule && tasks[i].schedule < opts->schedule) |
| 1309 | continue; |
| 1310 | |
Derrick Stolee | 25914c4 | 2020-09-17 18:11:52 +0000 | [diff] [blame] | 1311 | trace2_region_enter("maintenance", tasks[i].name, r); |
Derrick Stolee | 3103e98 | 2020-09-17 18:11:45 +0000 | [diff] [blame] | 1312 | if (tasks[i].fn(opts)) { |
| 1313 | error(_("task '%s' failed"), tasks[i].name); |
| 1314 | result = 1; |
| 1315 | } |
Derrick Stolee | 25914c4 | 2020-09-17 18:11:52 +0000 | [diff] [blame] | 1316 | trace2_region_leave("maintenance", tasks[i].name, r); |
Derrick Stolee | 3103e98 | 2020-09-17 18:11:45 +0000 | [diff] [blame] | 1317 | } |
| 1318 | |
Derrick Stolee | d7514f6 | 2020-09-17 18:11:48 +0000 | [diff] [blame] | 1319 | rollback_lock_file(&lk); |
Derrick Stolee | 3103e98 | 2020-09-17 18:11:45 +0000 | [diff] [blame] | 1320 | return result; |
| 1321 | } |
| 1322 | |
Derrick Stolee | a4cb1a2 | 2020-10-15 17:22:02 +0000 | [diff] [blame] | 1323 | static void initialize_maintenance_strategy(void) |
| 1324 | { |
| 1325 | char *config_str; |
| 1326 | |
| 1327 | if (git_config_get_string("maintenance.strategy", &config_str)) |
| 1328 | return; |
| 1329 | |
| 1330 | if (!strcasecmp(config_str, "incremental")) { |
| 1331 | tasks[TASK_GC].schedule = SCHEDULE_NONE; |
| 1332 | tasks[TASK_COMMIT_GRAPH].enabled = 1; |
| 1333 | tasks[TASK_COMMIT_GRAPH].schedule = SCHEDULE_HOURLY; |
| 1334 | tasks[TASK_PREFETCH].enabled = 1; |
| 1335 | tasks[TASK_PREFETCH].schedule = SCHEDULE_HOURLY; |
| 1336 | tasks[TASK_INCREMENTAL_REPACK].enabled = 1; |
| 1337 | tasks[TASK_INCREMENTAL_REPACK].schedule = SCHEDULE_DAILY; |
| 1338 | tasks[TASK_LOOSE_OBJECTS].enabled = 1; |
| 1339 | tasks[TASK_LOOSE_OBJECTS].schedule = SCHEDULE_DAILY; |
Derrick Stolee | acc1c4d | 2021-02-09 13:42:29 +0000 | [diff] [blame] | 1340 | tasks[TASK_PACK_REFS].enabled = 1; |
| 1341 | tasks[TASK_PACK_REFS].schedule = SCHEDULE_WEEKLY; |
Derrick Stolee | a4cb1a2 | 2020-10-15 17:22:02 +0000 | [diff] [blame] | 1342 | } |
| 1343 | } |
| 1344 | |
| 1345 | static void initialize_task_config(int schedule) |
Derrick Stolee | 65d655b | 2020-09-17 18:11:49 +0000 | [diff] [blame] | 1346 | { |
| 1347 | int i; |
| 1348 | struct strbuf config_name = STRBUF_INIT; |
Derrick Stolee | 916d062 | 2020-09-17 18:11:50 +0000 | [diff] [blame] | 1349 | gc_config(); |
| 1350 | |
Derrick Stolee | a4cb1a2 | 2020-10-15 17:22:02 +0000 | [diff] [blame] | 1351 | if (schedule) |
| 1352 | initialize_maintenance_strategy(); |
| 1353 | |
Derrick Stolee | 65d655b | 2020-09-17 18:11:49 +0000 | [diff] [blame] | 1354 | for (i = 0; i < TASK__COUNT; i++) { |
| 1355 | int config_value; |
Derrick Stolee | b08ff1f | 2020-09-11 17:49:15 +0000 | [diff] [blame] | 1356 | char *config_str; |
Derrick Stolee | 65d655b | 2020-09-17 18:11:49 +0000 | [diff] [blame] | 1357 | |
Derrick Stolee | b08ff1f | 2020-09-11 17:49:15 +0000 | [diff] [blame] | 1358 | strbuf_reset(&config_name); |
Derrick Stolee | 65d655b | 2020-09-17 18:11:49 +0000 | [diff] [blame] | 1359 | strbuf_addf(&config_name, "maintenance.%s.enabled", |
| 1360 | tasks[i].name); |
| 1361 | |
| 1362 | if (!git_config_get_bool(config_name.buf, &config_value)) |
| 1363 | tasks[i].enabled = config_value; |
Derrick Stolee | b08ff1f | 2020-09-11 17:49:15 +0000 | [diff] [blame] | 1364 | |
| 1365 | strbuf_reset(&config_name); |
| 1366 | strbuf_addf(&config_name, "maintenance.%s.schedule", |
| 1367 | tasks[i].name); |
| 1368 | |
| 1369 | if (!git_config_get_string(config_name.buf, &config_str)) { |
| 1370 | tasks[i].schedule = parse_schedule(config_str); |
| 1371 | free(config_str); |
| 1372 | } |
Derrick Stolee | 65d655b | 2020-09-17 18:11:49 +0000 | [diff] [blame] | 1373 | } |
| 1374 | |
| 1375 | strbuf_release(&config_name); |
| 1376 | } |
| 1377 | |
Derrick Stolee | 090511b | 2020-09-17 18:11:47 +0000 | [diff] [blame] | 1378 | static int task_option_parse(const struct option *opt, |
| 1379 | const char *arg, int unset) |
| 1380 | { |
| 1381 | int i, num_selected = 0; |
| 1382 | struct maintenance_task *task = NULL; |
| 1383 | |
| 1384 | BUG_ON_OPT_NEG(unset); |
| 1385 | |
| 1386 | for (i = 0; i < TASK__COUNT; i++) { |
| 1387 | if (tasks[i].selected_order >= 0) |
| 1388 | num_selected++; |
| 1389 | if (!strcasecmp(tasks[i].name, arg)) { |
| 1390 | task = &tasks[i]; |
| 1391 | } |
| 1392 | } |
| 1393 | |
| 1394 | if (!task) { |
| 1395 | error(_("'%s' is not a valid task"), arg); |
| 1396 | return 1; |
| 1397 | } |
| 1398 | |
| 1399 | if (task->selected_order >= 0) { |
| 1400 | error(_("task '%s' cannot be selected multiple times"), arg); |
| 1401 | return 1; |
| 1402 | } |
| 1403 | |
| 1404 | task->selected_order = num_selected + 1; |
| 1405 | |
| 1406 | return 0; |
| 1407 | } |
| 1408 | |
Derrick Stolee | 2057d75 | 2020-09-17 18:11:42 +0000 | [diff] [blame] | 1409 | static int maintenance_run(int argc, const char **argv, const char *prefix) |
| 1410 | { |
Derrick Stolee | 090511b | 2020-09-17 18:11:47 +0000 | [diff] [blame] | 1411 | int i; |
Derrick Stolee | 2057d75 | 2020-09-17 18:11:42 +0000 | [diff] [blame] | 1412 | struct maintenance_run_opts opts; |
| 1413 | struct option builtin_maintenance_run_options[] = { |
| 1414 | OPT_BOOL(0, "auto", &opts.auto_flag, |
| 1415 | N_("run tasks based on the state of the repository")), |
Derrick Stolee | b08ff1f | 2020-09-11 17:49:15 +0000 | [diff] [blame] | 1416 | OPT_CALLBACK(0, "schedule", &opts.schedule, N_("frequency"), |
| 1417 | N_("run tasks based on frequency"), |
| 1418 | maintenance_opt_schedule), |
Derrick Stolee | 3ddaad0 | 2020-09-17 18:11:43 +0000 | [diff] [blame] | 1419 | OPT_BOOL(0, "quiet", &opts.quiet, |
| 1420 | N_("do not report progress or other information over stderr")), |
Derrick Stolee | 090511b | 2020-09-17 18:11:47 +0000 | [diff] [blame] | 1421 | OPT_CALLBACK_F(0, "task", NULL, N_("task"), |
| 1422 | N_("run a specific task"), |
| 1423 | PARSE_OPT_NONEG, task_option_parse), |
Derrick Stolee | 2057d75 | 2020-09-17 18:11:42 +0000 | [diff] [blame] | 1424 | OPT_END() |
| 1425 | }; |
| 1426 | memset(&opts, 0, sizeof(opts)); |
| 1427 | |
Derrick Stolee | 3ddaad0 | 2020-09-17 18:11:43 +0000 | [diff] [blame] | 1428 | opts.quiet = !isatty(2); |
| 1429 | |
Derrick Stolee | 090511b | 2020-09-17 18:11:47 +0000 | [diff] [blame] | 1430 | for (i = 0; i < TASK__COUNT; i++) |
| 1431 | tasks[i].selected_order = -1; |
| 1432 | |
Derrick Stolee | 2057d75 | 2020-09-17 18:11:42 +0000 | [diff] [blame] | 1433 | argc = parse_options(argc, argv, prefix, |
| 1434 | builtin_maintenance_run_options, |
| 1435 | builtin_maintenance_run_usage, |
| 1436 | PARSE_OPT_STOP_AT_NON_OPTION); |
| 1437 | |
Derrick Stolee | b08ff1f | 2020-09-11 17:49:15 +0000 | [diff] [blame] | 1438 | if (opts.auto_flag && opts.schedule) |
| 1439 | die(_("use at most one of --auto and --schedule=<frequency>")); |
| 1440 | |
Derrick Stolee | a4cb1a2 | 2020-10-15 17:22:02 +0000 | [diff] [blame] | 1441 | initialize_task_config(opts.schedule); |
| 1442 | |
Derrick Stolee | 2057d75 | 2020-09-17 18:11:42 +0000 | [diff] [blame] | 1443 | if (argc != 0) |
| 1444 | usage_with_options(builtin_maintenance_run_usage, |
| 1445 | builtin_maintenance_run_options); |
Derrick Stolee | 3103e98 | 2020-09-17 18:11:45 +0000 | [diff] [blame] | 1446 | return maintenance_run_tasks(&opts); |
Derrick Stolee | 2057d75 | 2020-09-17 18:11:42 +0000 | [diff] [blame] | 1447 | } |
| 1448 | |
Eric Sunshine | 26c7974 | 2021-02-23 02:31:07 -0500 | [diff] [blame] | 1449 | static char *get_maintpath(void) |
| 1450 | { |
| 1451 | struct strbuf sb = STRBUF_INIT; |
| 1452 | const char *p = the_repository->worktree ? |
| 1453 | the_repository->worktree : the_repository->gitdir; |
| 1454 | |
| 1455 | strbuf_realpath(&sb, p, 1); |
| 1456 | return strbuf_detach(&sb, NULL); |
| 1457 | } |
| 1458 | |
Derrick Stolee | 0c18b70 | 2020-09-11 17:49:17 +0000 | [diff] [blame] | 1459 | static int maintenance_register(void) |
| 1460 | { |
Eric Sunshine | 26c7974 | 2021-02-23 02:31:07 -0500 | [diff] [blame] | 1461 | int rc; |
Derrick Stolee | 61f7a38 | 2020-10-15 17:22:03 +0000 | [diff] [blame] | 1462 | char *config_value; |
Derrick Stolee | 0c18b70 | 2020-09-11 17:49:17 +0000 | [diff] [blame] | 1463 | struct child_process config_set = CHILD_PROCESS_INIT; |
| 1464 | struct child_process config_get = CHILD_PROCESS_INIT; |
Eric Sunshine | 26c7974 | 2021-02-23 02:31:07 -0500 | [diff] [blame] | 1465 | char *maintpath = get_maintpath(); |
Derrick Stolee | 0c18b70 | 2020-09-11 17:49:17 +0000 | [diff] [blame] | 1466 | |
Derrick Stolee | 61f7a38 | 2020-10-15 17:22:03 +0000 | [diff] [blame] | 1467 | /* Disable foreground maintenance */ |
| 1468 | git_config_set("maintenance.auto", "false"); |
| 1469 | |
| 1470 | /* Set maintenance strategy, if unset */ |
| 1471 | if (!git_config_get_string("maintenance.strategy", &config_value)) |
| 1472 | free(config_value); |
| 1473 | else |
| 1474 | git_config_set("maintenance.strategy", "incremental"); |
| 1475 | |
Derrick Stolee | 0c18b70 | 2020-09-11 17:49:17 +0000 | [diff] [blame] | 1476 | config_get.git_cmd = 1; |
Derrick Stolee | 483a6d9 | 2020-11-25 22:12:56 +0000 | [diff] [blame] | 1477 | strvec_pushl(&config_get.args, "config", "--global", "--get", |
Eric Sunshine | 26c7974 | 2021-02-23 02:31:07 -0500 | [diff] [blame] | 1478 | "--fixed-value", "maintenance.repo", maintpath, NULL); |
Derrick Stolee | 0c18b70 | 2020-09-11 17:49:17 +0000 | [diff] [blame] | 1479 | config_get.out = -1; |
| 1480 | |
Eric Sunshine | 26c7974 | 2021-02-23 02:31:07 -0500 | [diff] [blame] | 1481 | if (start_command(&config_get)) { |
| 1482 | rc = error(_("failed to run 'git config'")); |
| 1483 | goto done; |
| 1484 | } |
Derrick Stolee | 0c18b70 | 2020-09-11 17:49:17 +0000 | [diff] [blame] | 1485 | |
| 1486 | /* We already have this value in our config! */ |
Eric Sunshine | 26c7974 | 2021-02-23 02:31:07 -0500 | [diff] [blame] | 1487 | if (!finish_command(&config_get)) { |
| 1488 | rc = 0; |
| 1489 | goto done; |
| 1490 | } |
Derrick Stolee | 0c18b70 | 2020-09-11 17:49:17 +0000 | [diff] [blame] | 1491 | |
| 1492 | config_set.git_cmd = 1; |
| 1493 | strvec_pushl(&config_set.args, "config", "--add", "--global", "maintenance.repo", |
Eric Sunshine | 26c7974 | 2021-02-23 02:31:07 -0500 | [diff] [blame] | 1494 | maintpath, NULL); |
Derrick Stolee | 0c18b70 | 2020-09-11 17:49:17 +0000 | [diff] [blame] | 1495 | |
Eric Sunshine | 26c7974 | 2021-02-23 02:31:07 -0500 | [diff] [blame] | 1496 | rc = run_command(&config_set); |
| 1497 | |
| 1498 | done: |
| 1499 | free(maintpath); |
| 1500 | return rc; |
Derrick Stolee | 0c18b70 | 2020-09-11 17:49:17 +0000 | [diff] [blame] | 1501 | } |
| 1502 | |
| 1503 | static int maintenance_unregister(void) |
| 1504 | { |
Eric Sunshine | 26c7974 | 2021-02-23 02:31:07 -0500 | [diff] [blame] | 1505 | int rc; |
Derrick Stolee | 0c18b70 | 2020-09-11 17:49:17 +0000 | [diff] [blame] | 1506 | struct child_process config_unset = CHILD_PROCESS_INIT; |
Eric Sunshine | 26c7974 | 2021-02-23 02:31:07 -0500 | [diff] [blame] | 1507 | char *maintpath = get_maintpath(); |
Derrick Stolee | 0c18b70 | 2020-09-11 17:49:17 +0000 | [diff] [blame] | 1508 | |
Derrick Stolee | 0c18b70 | 2020-09-11 17:49:17 +0000 | [diff] [blame] | 1509 | config_unset.git_cmd = 1; |
| 1510 | strvec_pushl(&config_unset.args, "config", "--global", "--unset", |
Eric Sunshine | 26c7974 | 2021-02-23 02:31:07 -0500 | [diff] [blame] | 1511 | "--fixed-value", "maintenance.repo", maintpath, NULL); |
Derrick Stolee | 0c18b70 | 2020-09-11 17:49:17 +0000 | [diff] [blame] | 1512 | |
Eric Sunshine | 26c7974 | 2021-02-23 02:31:07 -0500 | [diff] [blame] | 1513 | rc = run_command(&config_unset); |
| 1514 | free(maintpath); |
| 1515 | return rc; |
Derrick Stolee | 0c18b70 | 2020-09-11 17:49:17 +0000 | [diff] [blame] | 1516 | } |
| 1517 | |
Derrick Stolee | 2afe7e3 | 2021-01-05 13:08:27 +0000 | [diff] [blame] | 1518 | static const char *get_frequency(enum schedule_priority schedule) |
| 1519 | { |
| 1520 | switch (schedule) { |
| 1521 | case SCHEDULE_HOURLY: |
| 1522 | return "hourly"; |
| 1523 | case SCHEDULE_DAILY: |
| 1524 | return "daily"; |
| 1525 | case SCHEDULE_WEEKLY: |
| 1526 | return "weekly"; |
| 1527 | default: |
| 1528 | BUG("invalid schedule %d", schedule); |
| 1529 | } |
| 1530 | } |
| 1531 | |
| 1532 | static char *launchctl_service_name(const char *frequency) |
| 1533 | { |
| 1534 | struct strbuf label = STRBUF_INIT; |
| 1535 | strbuf_addf(&label, "org.git-scm.git.%s", frequency); |
| 1536 | return strbuf_detach(&label, NULL); |
| 1537 | } |
| 1538 | |
| 1539 | static char *launchctl_service_filename(const char *name) |
| 1540 | { |
| 1541 | char *expanded; |
| 1542 | struct strbuf filename = STRBUF_INIT; |
| 1543 | strbuf_addf(&filename, "~/Library/LaunchAgents/%s.plist", name); |
| 1544 | |
Johannes Schindelin | a03b097 | 2021-07-24 22:06:52 +0000 | [diff] [blame] | 1545 | expanded = interpolate_path(filename.buf, 1); |
Derrick Stolee | 2afe7e3 | 2021-01-05 13:08:27 +0000 | [diff] [blame] | 1546 | if (!expanded) |
| 1547 | die(_("failed to expand path '%s'"), filename.buf); |
| 1548 | |
| 1549 | strbuf_release(&filename); |
| 1550 | return expanded; |
| 1551 | } |
| 1552 | |
| 1553 | static char *launchctl_get_uid(void) |
| 1554 | { |
| 1555 | return xstrfmt("gui/%d", getuid()); |
| 1556 | } |
| 1557 | |
| 1558 | static int launchctl_boot_plist(int enable, const char *filename, const char *cmd) |
| 1559 | { |
| 1560 | int result; |
| 1561 | struct child_process child = CHILD_PROCESS_INIT; |
| 1562 | char *uid = launchctl_get_uid(); |
| 1563 | |
| 1564 | strvec_split(&child.args, cmd); |
| 1565 | if (enable) |
| 1566 | strvec_push(&child.args, "bootstrap"); |
| 1567 | else |
| 1568 | strvec_push(&child.args, "bootout"); |
| 1569 | strvec_push(&child.args, uid); |
| 1570 | strvec_push(&child.args, filename); |
| 1571 | |
| 1572 | child.no_stderr = 1; |
| 1573 | child.no_stdout = 1; |
| 1574 | |
| 1575 | if (start_command(&child)) |
| 1576 | die(_("failed to start launchctl")); |
| 1577 | |
| 1578 | result = finish_command(&child); |
| 1579 | |
| 1580 | free(uid); |
| 1581 | return result; |
| 1582 | } |
| 1583 | |
| 1584 | static int launchctl_remove_plist(enum schedule_priority schedule, const char *cmd) |
| 1585 | { |
| 1586 | const char *frequency = get_frequency(schedule); |
| 1587 | char *name = launchctl_service_name(frequency); |
| 1588 | char *filename = launchctl_service_filename(name); |
| 1589 | int result = launchctl_boot_plist(0, filename, cmd); |
| 1590 | unlink(filename); |
| 1591 | free(filename); |
| 1592 | free(name); |
| 1593 | return result; |
| 1594 | } |
| 1595 | |
| 1596 | static int launchctl_remove_plists(const char *cmd) |
| 1597 | { |
| 1598 | return launchctl_remove_plist(SCHEDULE_HOURLY, cmd) || |
| 1599 | launchctl_remove_plist(SCHEDULE_DAILY, cmd) || |
| 1600 | launchctl_remove_plist(SCHEDULE_WEEKLY, cmd); |
| 1601 | } |
| 1602 | |
Derrick Stolee | a16eb6b | 2021-08-24 15:44:00 +0000 | [diff] [blame] | 1603 | static int launchctl_list_contains_plist(const char *name, const char *cmd) |
| 1604 | { |
Derrick Stolee | a16eb6b | 2021-08-24 15:44:00 +0000 | [diff] [blame] | 1605 | struct child_process child = CHILD_PROCESS_INIT; |
Derrick Stolee | a16eb6b | 2021-08-24 15:44:00 +0000 | [diff] [blame] | 1606 | |
| 1607 | strvec_split(&child.args, cmd); |
| 1608 | strvec_pushl(&child.args, "list", name, NULL); |
| 1609 | |
| 1610 | child.no_stderr = 1; |
| 1611 | child.no_stdout = 1; |
| 1612 | |
| 1613 | if (start_command(&child)) |
| 1614 | die(_("failed to start launchctl")); |
| 1615 | |
Derrick Stolee | a16eb6b | 2021-08-24 15:44:00 +0000 | [diff] [blame] | 1616 | /* Returns failure if 'name' doesn't exist. */ |
Ævar Arnfjörð Bjarmason | 3218cb7 | 2021-09-12 02:24:40 +0200 | [diff] [blame^] | 1617 | return !finish_command(&child); |
Derrick Stolee | a16eb6b | 2021-08-24 15:44:00 +0000 | [diff] [blame] | 1618 | } |
| 1619 | |
Derrick Stolee | 2afe7e3 | 2021-01-05 13:08:27 +0000 | [diff] [blame] | 1620 | static int launchctl_schedule_plist(const char *exec_path, enum schedule_priority schedule, const char *cmd) |
| 1621 | { |
Johannes Schindelin | bb01122 | 2021-08-24 15:43:59 +0000 | [diff] [blame] | 1622 | int i, fd; |
Derrick Stolee | 2afe7e3 | 2021-01-05 13:08:27 +0000 | [diff] [blame] | 1623 | const char *preamble, *repeat; |
| 1624 | const char *frequency = get_frequency(schedule); |
| 1625 | char *name = launchctl_service_name(frequency); |
| 1626 | char *filename = launchctl_service_filename(name); |
Johannes Schindelin | bb01122 | 2021-08-24 15:43:59 +0000 | [diff] [blame] | 1627 | struct lock_file lk = LOCK_INIT; |
| 1628 | static unsigned long lock_file_timeout_ms = ULONG_MAX; |
Derrick Stolee | a16eb6b | 2021-08-24 15:44:00 +0000 | [diff] [blame] | 1629 | struct strbuf plist = STRBUF_INIT, plist2 = STRBUF_INIT; |
| 1630 | struct stat st; |
Derrick Stolee | 2afe7e3 | 2021-01-05 13:08:27 +0000 | [diff] [blame] | 1631 | |
Derrick Stolee | 3797a0a | 2021-01-05 13:08:28 +0000 | [diff] [blame] | 1632 | preamble = "<?xml version=\"1.0\"?>\n" |
Derrick Stolee | 2afe7e3 | 2021-01-05 13:08:27 +0000 | [diff] [blame] | 1633 | "<!DOCTYPE plist PUBLIC \"-//Apple//DTD PLIST 1.0//EN\" \"http://www.apple.com/DTDs/PropertyList-1.0.dtd\">\n" |
| 1634 | "<plist version=\"1.0\">" |
| 1635 | "<dict>\n" |
| 1636 | "<key>Label</key><string>%s</string>\n" |
| 1637 | "<key>ProgramArguments</key>\n" |
| 1638 | "<array>\n" |
| 1639 | "<string>%s/git</string>\n" |
| 1640 | "<string>--exec-path=%s</string>\n" |
| 1641 | "<string>for-each-repo</string>\n" |
| 1642 | "<string>--config=maintenance.repo</string>\n" |
| 1643 | "<string>maintenance</string>\n" |
| 1644 | "<string>run</string>\n" |
| 1645 | "<string>--schedule=%s</string>\n" |
| 1646 | "</array>\n" |
| 1647 | "<key>StartCalendarInterval</key>\n" |
| 1648 | "<array>\n"; |
Johannes Schindelin | bb01122 | 2021-08-24 15:43:59 +0000 | [diff] [blame] | 1649 | strbuf_addf(&plist, preamble, name, exec_path, exec_path, frequency); |
Derrick Stolee | 2afe7e3 | 2021-01-05 13:08:27 +0000 | [diff] [blame] | 1650 | |
| 1651 | switch (schedule) { |
| 1652 | case SCHEDULE_HOURLY: |
| 1653 | repeat = "<dict>\n" |
| 1654 | "<key>Hour</key><integer>%d</integer>\n" |
| 1655 | "<key>Minute</key><integer>0</integer>\n" |
| 1656 | "</dict>\n"; |
| 1657 | for (i = 1; i <= 23; i++) |
Johannes Schindelin | bb01122 | 2021-08-24 15:43:59 +0000 | [diff] [blame] | 1658 | strbuf_addf(&plist, repeat, i); |
Derrick Stolee | 2afe7e3 | 2021-01-05 13:08:27 +0000 | [diff] [blame] | 1659 | break; |
| 1660 | |
| 1661 | case SCHEDULE_DAILY: |
| 1662 | repeat = "<dict>\n" |
| 1663 | "<key>Day</key><integer>%d</integer>\n" |
| 1664 | "<key>Hour</key><integer>0</integer>\n" |
| 1665 | "<key>Minute</key><integer>0</integer>\n" |
| 1666 | "</dict>\n"; |
| 1667 | for (i = 1; i <= 6; i++) |
Johannes Schindelin | bb01122 | 2021-08-24 15:43:59 +0000 | [diff] [blame] | 1668 | strbuf_addf(&plist, repeat, i); |
Derrick Stolee | 2afe7e3 | 2021-01-05 13:08:27 +0000 | [diff] [blame] | 1669 | break; |
| 1670 | |
| 1671 | case SCHEDULE_WEEKLY: |
Johannes Schindelin | bb01122 | 2021-08-24 15:43:59 +0000 | [diff] [blame] | 1672 | strbuf_addstr(&plist, |
| 1673 | "<dict>\n" |
| 1674 | "<key>Day</key><integer>0</integer>\n" |
| 1675 | "<key>Hour</key><integer>0</integer>\n" |
| 1676 | "<key>Minute</key><integer>0</integer>\n" |
| 1677 | "</dict>\n"); |
Derrick Stolee | 2afe7e3 | 2021-01-05 13:08:27 +0000 | [diff] [blame] | 1678 | break; |
| 1679 | |
| 1680 | default: |
| 1681 | /* unreachable */ |
| 1682 | break; |
| 1683 | } |
Johannes Schindelin | bb01122 | 2021-08-24 15:43:59 +0000 | [diff] [blame] | 1684 | strbuf_addstr(&plist, "</array>\n</dict>\n</plist>\n"); |
Derrick Stolee | 2afe7e3 | 2021-01-05 13:08:27 +0000 | [diff] [blame] | 1685 | |
Johannes Schindelin | bb01122 | 2021-08-24 15:43:59 +0000 | [diff] [blame] | 1686 | if (safe_create_leading_directories(filename)) |
| 1687 | die(_("failed to create directories for '%s'"), filename); |
| 1688 | |
| 1689 | if ((long)lock_file_timeout_ms < 0 && |
| 1690 | git_config_get_ulong("gc.launchctlplistlocktimeoutms", |
| 1691 | &lock_file_timeout_ms)) |
| 1692 | lock_file_timeout_ms = 150; |
| 1693 | |
| 1694 | fd = hold_lock_file_for_update_timeout(&lk, filename, LOCK_DIE_ON_ERROR, |
| 1695 | lock_file_timeout_ms); |
| 1696 | |
Derrick Stolee | a16eb6b | 2021-08-24 15:44:00 +0000 | [diff] [blame] | 1697 | /* |
| 1698 | * Does this file already exist? With the intended contents? Is it |
| 1699 | * registered already? Then it does not need to be re-registered. |
| 1700 | */ |
| 1701 | if (!stat(filename, &st) && st.st_size == plist.len && |
| 1702 | strbuf_read_file(&plist2, filename, plist.len) == plist.len && |
| 1703 | !strbuf_cmp(&plist, &plist2) && |
| 1704 | launchctl_list_contains_plist(name, cmd)) |
| 1705 | rollback_lock_file(&lk); |
| 1706 | else { |
| 1707 | if (write_in_full(fd, plist.buf, plist.len) < 0 || |
| 1708 | commit_lock_file(&lk)) |
| 1709 | die_errno(_("could not write '%s'"), filename); |
Derrick Stolee | 2afe7e3 | 2021-01-05 13:08:27 +0000 | [diff] [blame] | 1710 | |
Derrick Stolee | a16eb6b | 2021-08-24 15:44:00 +0000 | [diff] [blame] | 1711 | /* bootout might fail if not already running, so ignore */ |
| 1712 | launchctl_boot_plist(0, filename, cmd); |
| 1713 | if (launchctl_boot_plist(1, filename, cmd)) |
| 1714 | die(_("failed to bootstrap service %s"), filename); |
| 1715 | } |
Derrick Stolee | 2afe7e3 | 2021-01-05 13:08:27 +0000 | [diff] [blame] | 1716 | |
| 1717 | free(filename); |
| 1718 | free(name); |
Johannes Schindelin | bb01122 | 2021-08-24 15:43:59 +0000 | [diff] [blame] | 1719 | strbuf_release(&plist); |
Derrick Stolee | a16eb6b | 2021-08-24 15:44:00 +0000 | [diff] [blame] | 1720 | strbuf_release(&plist2); |
Derrick Stolee | 2afe7e3 | 2021-01-05 13:08:27 +0000 | [diff] [blame] | 1721 | return 0; |
| 1722 | } |
| 1723 | |
| 1724 | static int launchctl_add_plists(const char *cmd) |
| 1725 | { |
| 1726 | const char *exec_path = git_exec_path(); |
| 1727 | |
| 1728 | return launchctl_schedule_plist(exec_path, SCHEDULE_HOURLY, cmd) || |
| 1729 | launchctl_schedule_plist(exec_path, SCHEDULE_DAILY, cmd) || |
| 1730 | launchctl_schedule_plist(exec_path, SCHEDULE_WEEKLY, cmd); |
| 1731 | } |
| 1732 | |
| 1733 | static int launchctl_update_schedule(int run_maintenance, int fd, const char *cmd) |
| 1734 | { |
| 1735 | if (run_maintenance) |
| 1736 | return launchctl_add_plists(cmd); |
| 1737 | else |
| 1738 | return launchctl_remove_plists(cmd); |
| 1739 | } |
| 1740 | |
Derrick Stolee | 3797a0a | 2021-01-05 13:08:28 +0000 | [diff] [blame] | 1741 | static char *schtasks_task_name(const char *frequency) |
| 1742 | { |
| 1743 | struct strbuf label = STRBUF_INIT; |
| 1744 | strbuf_addf(&label, "Git Maintenance (%s)", frequency); |
| 1745 | return strbuf_detach(&label, NULL); |
| 1746 | } |
| 1747 | |
| 1748 | static int schtasks_remove_task(enum schedule_priority schedule, const char *cmd) |
| 1749 | { |
| 1750 | int result; |
| 1751 | struct strvec args = STRVEC_INIT; |
| 1752 | const char *frequency = get_frequency(schedule); |
| 1753 | char *name = schtasks_task_name(frequency); |
| 1754 | |
| 1755 | strvec_split(&args, cmd); |
| 1756 | strvec_pushl(&args, "/delete", "/tn", name, "/f", NULL); |
| 1757 | |
| 1758 | result = run_command_v_opt(args.v, 0); |
| 1759 | |
| 1760 | strvec_clear(&args); |
| 1761 | free(name); |
| 1762 | return result; |
| 1763 | } |
| 1764 | |
| 1765 | static int schtasks_remove_tasks(const char *cmd) |
| 1766 | { |
| 1767 | return schtasks_remove_task(SCHEDULE_HOURLY, cmd) || |
| 1768 | schtasks_remove_task(SCHEDULE_DAILY, cmd) || |
| 1769 | schtasks_remove_task(SCHEDULE_WEEKLY, cmd); |
| 1770 | } |
| 1771 | |
| 1772 | static int schtasks_schedule_task(const char *exec_path, enum schedule_priority schedule, const char *cmd) |
| 1773 | { |
| 1774 | int result; |
| 1775 | struct child_process child = CHILD_PROCESS_INIT; |
| 1776 | const char *xml; |
| 1777 | struct tempfile *tfile; |
| 1778 | const char *frequency = get_frequency(schedule); |
| 1779 | char *name = schtasks_task_name(frequency); |
| 1780 | struct strbuf tfilename = STRBUF_INIT; |
| 1781 | |
| 1782 | strbuf_addf(&tfilename, "%s/schedule_%s_XXXXXX", |
| 1783 | get_git_common_dir(), frequency); |
| 1784 | tfile = xmks_tempfile(tfilename.buf); |
| 1785 | strbuf_release(&tfilename); |
| 1786 | |
| 1787 | if (!fdopen_tempfile(tfile, "w")) |
| 1788 | die(_("failed to create temp xml file")); |
| 1789 | |
| 1790 | xml = "<?xml version=\"1.0\" ?>\n" |
| 1791 | "<Task version=\"1.4\" xmlns=\"http://schemas.microsoft.com/windows/2004/02/mit/task\">\n" |
| 1792 | "<Triggers>\n" |
| 1793 | "<CalendarTrigger>\n"; |
| 1794 | fputs(xml, tfile->fp); |
| 1795 | |
| 1796 | switch (schedule) { |
| 1797 | case SCHEDULE_HOURLY: |
| 1798 | fprintf(tfile->fp, |
| 1799 | "<StartBoundary>2020-01-01T01:00:00</StartBoundary>\n" |
| 1800 | "<Enabled>true</Enabled>\n" |
| 1801 | "<ScheduleByDay>\n" |
| 1802 | "<DaysInterval>1</DaysInterval>\n" |
| 1803 | "</ScheduleByDay>\n" |
| 1804 | "<Repetition>\n" |
| 1805 | "<Interval>PT1H</Interval>\n" |
| 1806 | "<Duration>PT23H</Duration>\n" |
| 1807 | "<StopAtDurationEnd>false</StopAtDurationEnd>\n" |
| 1808 | "</Repetition>\n"); |
| 1809 | break; |
| 1810 | |
| 1811 | case SCHEDULE_DAILY: |
| 1812 | fprintf(tfile->fp, |
| 1813 | "<StartBoundary>2020-01-01T00:00:00</StartBoundary>\n" |
| 1814 | "<Enabled>true</Enabled>\n" |
| 1815 | "<ScheduleByWeek>\n" |
| 1816 | "<DaysOfWeek>\n" |
| 1817 | "<Monday />\n" |
| 1818 | "<Tuesday />\n" |
| 1819 | "<Wednesday />\n" |
| 1820 | "<Thursday />\n" |
| 1821 | "<Friday />\n" |
| 1822 | "<Saturday />\n" |
| 1823 | "</DaysOfWeek>\n" |
| 1824 | "<WeeksInterval>1</WeeksInterval>\n" |
| 1825 | "</ScheduleByWeek>\n"); |
| 1826 | break; |
| 1827 | |
| 1828 | case SCHEDULE_WEEKLY: |
| 1829 | fprintf(tfile->fp, |
| 1830 | "<StartBoundary>2020-01-01T00:00:00</StartBoundary>\n" |
| 1831 | "<Enabled>true</Enabled>\n" |
| 1832 | "<ScheduleByWeek>\n" |
| 1833 | "<DaysOfWeek>\n" |
| 1834 | "<Sunday />\n" |
| 1835 | "</DaysOfWeek>\n" |
| 1836 | "<WeeksInterval>1</WeeksInterval>\n" |
| 1837 | "</ScheduleByWeek>\n"); |
| 1838 | break; |
| 1839 | |
| 1840 | default: |
| 1841 | break; |
| 1842 | } |
| 1843 | |
| 1844 | xml = "</CalendarTrigger>\n" |
| 1845 | "</Triggers>\n" |
| 1846 | "<Principals>\n" |
| 1847 | "<Principal id=\"Author\">\n" |
| 1848 | "<LogonType>InteractiveToken</LogonType>\n" |
| 1849 | "<RunLevel>LeastPrivilege</RunLevel>\n" |
| 1850 | "</Principal>\n" |
| 1851 | "</Principals>\n" |
| 1852 | "<Settings>\n" |
| 1853 | "<MultipleInstancesPolicy>IgnoreNew</MultipleInstancesPolicy>\n" |
| 1854 | "<Enabled>true</Enabled>\n" |
| 1855 | "<Hidden>true</Hidden>\n" |
| 1856 | "<UseUnifiedSchedulingEngine>true</UseUnifiedSchedulingEngine>\n" |
| 1857 | "<WakeToRun>false</WakeToRun>\n" |
| 1858 | "<ExecutionTimeLimit>PT72H</ExecutionTimeLimit>\n" |
| 1859 | "<Priority>7</Priority>\n" |
| 1860 | "</Settings>\n" |
| 1861 | "<Actions Context=\"Author\">\n" |
| 1862 | "<Exec>\n" |
| 1863 | "<Command>\"%s\\git.exe\"</Command>\n" |
| 1864 | "<Arguments>--exec-path=\"%s\" for-each-repo --config=maintenance.repo maintenance run --schedule=%s</Arguments>\n" |
| 1865 | "</Exec>\n" |
| 1866 | "</Actions>\n" |
| 1867 | "</Task>\n"; |
| 1868 | fprintf(tfile->fp, xml, exec_path, exec_path, frequency); |
| 1869 | strvec_split(&child.args, cmd); |
| 1870 | strvec_pushl(&child.args, "/create", "/tn", name, "/f", "/xml", |
| 1871 | get_tempfile_path(tfile), NULL); |
| 1872 | close_tempfile_gently(tfile); |
| 1873 | |
| 1874 | child.no_stdout = 1; |
| 1875 | child.no_stderr = 1; |
| 1876 | |
| 1877 | if (start_command(&child)) |
| 1878 | die(_("failed to start schtasks")); |
| 1879 | result = finish_command(&child); |
| 1880 | |
| 1881 | delete_tempfile(&tfile); |
| 1882 | free(name); |
| 1883 | return result; |
| 1884 | } |
| 1885 | |
| 1886 | static int schtasks_schedule_tasks(const char *cmd) |
| 1887 | { |
| 1888 | const char *exec_path = git_exec_path(); |
| 1889 | |
| 1890 | return schtasks_schedule_task(exec_path, SCHEDULE_HOURLY, cmd) || |
| 1891 | schtasks_schedule_task(exec_path, SCHEDULE_DAILY, cmd) || |
| 1892 | schtasks_schedule_task(exec_path, SCHEDULE_WEEKLY, cmd); |
| 1893 | } |
| 1894 | |
| 1895 | static int schtasks_update_schedule(int run_maintenance, int fd, const char *cmd) |
| 1896 | { |
| 1897 | if (run_maintenance) |
| 1898 | return schtasks_schedule_tasks(cmd); |
| 1899 | else |
| 1900 | return schtasks_remove_tasks(cmd); |
| 1901 | } |
| 1902 | |
Derrick Stolee | 2fec604 | 2020-09-11 17:49:18 +0000 | [diff] [blame] | 1903 | #define BEGIN_LINE "# BEGIN GIT MAINTENANCE SCHEDULE" |
| 1904 | #define END_LINE "# END GIT MAINTENANCE SCHEDULE" |
| 1905 | |
Derrick Stolee | 31345d5 | 2020-11-24 04:16:42 +0000 | [diff] [blame] | 1906 | static int crontab_update_schedule(int run_maintenance, int fd, const char *cmd) |
Derrick Stolee | 2fec604 | 2020-09-11 17:49:18 +0000 | [diff] [blame] | 1907 | { |
| 1908 | int result = 0; |
| 1909 | int in_old_region = 0; |
| 1910 | struct child_process crontab_list = CHILD_PROCESS_INIT; |
| 1911 | struct child_process crontab_edit = CHILD_PROCESS_INIT; |
| 1912 | FILE *cron_list, *cron_in; |
Derrick Stolee | 2fec604 | 2020-09-11 17:49:18 +0000 | [diff] [blame] | 1913 | struct strbuf line = STRBUF_INIT; |
Derrick Stolee | 2fec604 | 2020-09-11 17:49:18 +0000 | [diff] [blame] | 1914 | |
Derrick Stolee | 31345d5 | 2020-11-24 04:16:42 +0000 | [diff] [blame] | 1915 | strvec_split(&crontab_list.args, cmd); |
Derrick Stolee | 2fec604 | 2020-09-11 17:49:18 +0000 | [diff] [blame] | 1916 | strvec_push(&crontab_list.args, "-l"); |
| 1917 | crontab_list.in = -1; |
Derrick Stolee | 31345d5 | 2020-11-24 04:16:42 +0000 | [diff] [blame] | 1918 | crontab_list.out = dup(fd); |
Derrick Stolee | 2fec604 | 2020-09-11 17:49:18 +0000 | [diff] [blame] | 1919 | crontab_list.git_cmd = 0; |
| 1920 | |
Derrick Stolee | 31345d5 | 2020-11-24 04:16:42 +0000 | [diff] [blame] | 1921 | if (start_command(&crontab_list)) |
| 1922 | return error(_("failed to run 'crontab -l'; your system might not support 'cron'")); |
Derrick Stolee | 2fec604 | 2020-09-11 17:49:18 +0000 | [diff] [blame] | 1923 | |
| 1924 | /* Ignore exit code, as an empty crontab will return error. */ |
| 1925 | finish_command(&crontab_list); |
| 1926 | |
| 1927 | /* |
| 1928 | * Read from the .lock file, filtering out the old |
| 1929 | * schedule while appending the new schedule. |
| 1930 | */ |
Derrick Stolee | 31345d5 | 2020-11-24 04:16:42 +0000 | [diff] [blame] | 1931 | cron_list = fdopen(fd, "r"); |
Derrick Stolee | 2fec604 | 2020-09-11 17:49:18 +0000 | [diff] [blame] | 1932 | rewind(cron_list); |
| 1933 | |
Derrick Stolee | 31345d5 | 2020-11-24 04:16:42 +0000 | [diff] [blame] | 1934 | strvec_split(&crontab_edit.args, cmd); |
Derrick Stolee | 2fec604 | 2020-09-11 17:49:18 +0000 | [diff] [blame] | 1935 | crontab_edit.in = -1; |
| 1936 | crontab_edit.git_cmd = 0; |
| 1937 | |
Derrick Stolee | 31345d5 | 2020-11-24 04:16:42 +0000 | [diff] [blame] | 1938 | if (start_command(&crontab_edit)) |
| 1939 | return error(_("failed to run 'crontab'; your system might not support 'cron'")); |
Derrick Stolee | 2fec604 | 2020-09-11 17:49:18 +0000 | [diff] [blame] | 1940 | |
| 1941 | cron_in = fdopen(crontab_edit.in, "w"); |
| 1942 | if (!cron_in) { |
| 1943 | result = error(_("failed to open stdin of 'crontab'")); |
| 1944 | goto done_editing; |
| 1945 | } |
| 1946 | |
| 1947 | while (!strbuf_getline_lf(&line, cron_list)) { |
| 1948 | if (!in_old_region && !strcmp(line.buf, BEGIN_LINE)) |
| 1949 | in_old_region = 1; |
Martin Ågren | 66dc0a3 | 2020-12-21 22:26:32 +0100 | [diff] [blame] | 1950 | else if (in_old_region && !strcmp(line.buf, END_LINE)) |
Derrick Stolee | 2fec604 | 2020-09-11 17:49:18 +0000 | [diff] [blame] | 1951 | in_old_region = 0; |
Martin Ågren | 66dc0a3 | 2020-12-21 22:26:32 +0100 | [diff] [blame] | 1952 | else if (!in_old_region) |
| 1953 | fprintf(cron_in, "%s\n", line.buf); |
Derrick Stolee | 2fec604 | 2020-09-11 17:49:18 +0000 | [diff] [blame] | 1954 | } |
Lénaïc Huard | c5d0b12 | 2021-05-10 21:59:09 +0200 | [diff] [blame] | 1955 | strbuf_release(&line); |
Derrick Stolee | 2fec604 | 2020-09-11 17:49:18 +0000 | [diff] [blame] | 1956 | |
| 1957 | if (run_maintenance) { |
| 1958 | struct strbuf line_format = STRBUF_INIT; |
| 1959 | const char *exec_path = git_exec_path(); |
| 1960 | |
| 1961 | fprintf(cron_in, "%s\n", BEGIN_LINE); |
| 1962 | fprintf(cron_in, |
| 1963 | "# The following schedule was created by Git\n"); |
| 1964 | fprintf(cron_in, "# Any edits made in this region might be\n"); |
| 1965 | fprintf(cron_in, |
| 1966 | "# replaced in the future by a Git command.\n\n"); |
| 1967 | |
| 1968 | strbuf_addf(&line_format, |
| 1969 | "%%s %%s * * %%s \"%s/git\" --exec-path=\"%s\" for-each-repo --config=maintenance.repo maintenance run --schedule=%%s\n", |
| 1970 | exec_path, exec_path); |
| 1971 | fprintf(cron_in, line_format.buf, "0", "1-23", "*", "hourly"); |
| 1972 | fprintf(cron_in, line_format.buf, "0", "0", "1-6", "daily"); |
| 1973 | fprintf(cron_in, line_format.buf, "0", "0", "0", "weekly"); |
| 1974 | strbuf_release(&line_format); |
| 1975 | |
| 1976 | fprintf(cron_in, "\n%s\n", END_LINE); |
| 1977 | } |
| 1978 | |
| 1979 | fflush(cron_in); |
| 1980 | fclose(cron_in); |
| 1981 | close(crontab_edit.in); |
| 1982 | |
| 1983 | done_editing: |
Derrick Stolee | 31345d5 | 2020-11-24 04:16:42 +0000 | [diff] [blame] | 1984 | if (finish_command(&crontab_edit)) |
Derrick Stolee | 2fec604 | 2020-09-11 17:49:18 +0000 | [diff] [blame] | 1985 | result = error(_("'crontab' died")); |
Derrick Stolee | 31345d5 | 2020-11-24 04:16:42 +0000 | [diff] [blame] | 1986 | else |
| 1987 | fclose(cron_list); |
| 1988 | return result; |
| 1989 | } |
Derrick Stolee | 2fec604 | 2020-09-11 17:49:18 +0000 | [diff] [blame] | 1990 | |
Derrick Stolee | 2afe7e3 | 2021-01-05 13:08:27 +0000 | [diff] [blame] | 1991 | #if defined(__APPLE__) |
| 1992 | static const char platform_scheduler[] = "launchctl"; |
Derrick Stolee | 3797a0a | 2021-01-05 13:08:28 +0000 | [diff] [blame] | 1993 | #elif defined(GIT_WINDOWS_NATIVE) |
| 1994 | static const char platform_scheduler[] = "schtasks"; |
Derrick Stolee | 2afe7e3 | 2021-01-05 13:08:27 +0000 | [diff] [blame] | 1995 | #else |
Derrick Stolee | 31345d5 | 2020-11-24 04:16:42 +0000 | [diff] [blame] | 1996 | static const char platform_scheduler[] = "crontab"; |
Derrick Stolee | 2afe7e3 | 2021-01-05 13:08:27 +0000 | [diff] [blame] | 1997 | #endif |
Derrick Stolee | 31345d5 | 2020-11-24 04:16:42 +0000 | [diff] [blame] | 1998 | |
| 1999 | static int update_background_schedule(int enable) |
| 2000 | { |
| 2001 | int result; |
| 2002 | const char *scheduler = platform_scheduler; |
| 2003 | const char *cmd = scheduler; |
| 2004 | char *testing; |
| 2005 | struct lock_file lk; |
| 2006 | char *lock_path = xstrfmt("%s/schedule", the_repository->objects->odb->path); |
| 2007 | |
| 2008 | testing = xstrdup_or_null(getenv("GIT_TEST_MAINT_SCHEDULER")); |
| 2009 | if (testing) { |
| 2010 | char *sep = strchr(testing, ':'); |
| 2011 | if (!sep) |
| 2012 | die("GIT_TEST_MAINT_SCHEDULER unparseable: %s", testing); |
| 2013 | *sep = '\0'; |
| 2014 | scheduler = testing; |
| 2015 | cmd = sep + 1; |
| 2016 | } |
| 2017 | |
Lénaïc Huard | c5d0b12 | 2021-05-10 21:59:09 +0200 | [diff] [blame] | 2018 | if (hold_lock_file_for_update(&lk, lock_path, LOCK_NO_DEREF) < 0) { |
| 2019 | result = error(_("another process is scheduling background maintenance")); |
| 2020 | goto cleanup; |
| 2021 | } |
Derrick Stolee | 31345d5 | 2020-11-24 04:16:42 +0000 | [diff] [blame] | 2022 | |
Derrick Stolee | 2afe7e3 | 2021-01-05 13:08:27 +0000 | [diff] [blame] | 2023 | if (!strcmp(scheduler, "launchctl")) |
Junio C Hamano | 58e2ce9 | 2021-01-25 14:19:17 -0800 | [diff] [blame] | 2024 | result = launchctl_update_schedule(enable, get_lock_file_fd(&lk), cmd); |
Derrick Stolee | 3797a0a | 2021-01-05 13:08:28 +0000 | [diff] [blame] | 2025 | else if (!strcmp(scheduler, "schtasks")) |
Junio C Hamano | 58e2ce9 | 2021-01-25 14:19:17 -0800 | [diff] [blame] | 2026 | result = schtasks_update_schedule(enable, get_lock_file_fd(&lk), cmd); |
Derrick Stolee | 2afe7e3 | 2021-01-05 13:08:27 +0000 | [diff] [blame] | 2027 | else if (!strcmp(scheduler, "crontab")) |
Junio C Hamano | 58e2ce9 | 2021-01-25 14:19:17 -0800 | [diff] [blame] | 2028 | result = crontab_update_schedule(enable, get_lock_file_fd(&lk), cmd); |
Derrick Stolee | 31345d5 | 2020-11-24 04:16:42 +0000 | [diff] [blame] | 2029 | else |
| 2030 | die("unknown background scheduler: %s", scheduler); |
| 2031 | |
Derrick Stolee | 2fec604 | 2020-09-11 17:49:18 +0000 | [diff] [blame] | 2032 | rollback_lock_file(&lk); |
Lénaïc Huard | c5d0b12 | 2021-05-10 21:59:09 +0200 | [diff] [blame] | 2033 | |
| 2034 | cleanup: |
| 2035 | free(lock_path); |
Derrick Stolee | 31345d5 | 2020-11-24 04:16:42 +0000 | [diff] [blame] | 2036 | free(testing); |
Derrick Stolee | 2fec604 | 2020-09-11 17:49:18 +0000 | [diff] [blame] | 2037 | return result; |
| 2038 | } |
| 2039 | |
| 2040 | static int maintenance_start(void) |
| 2041 | { |
| 2042 | if (maintenance_register()) |
| 2043 | warning(_("failed to add repo to global config")); |
| 2044 | |
| 2045 | return update_background_schedule(1); |
| 2046 | } |
| 2047 | |
| 2048 | static int maintenance_stop(void) |
| 2049 | { |
| 2050 | return update_background_schedule(0); |
| 2051 | } |
| 2052 | |
Derrick Stolee | 0c18b70 | 2020-09-11 17:49:17 +0000 | [diff] [blame] | 2053 | static const char builtin_maintenance_usage[] = N_("git maintenance <subcommand> [<options>]"); |
Derrick Stolee | 2057d75 | 2020-09-17 18:11:42 +0000 | [diff] [blame] | 2054 | |
| 2055 | int cmd_maintenance(int argc, const char **argv, const char *prefix) |
| 2056 | { |
| 2057 | if (argc < 2 || |
| 2058 | (argc == 2 && !strcmp(argv[1], "-h"))) |
| 2059 | usage(builtin_maintenance_usage); |
| 2060 | |
| 2061 | if (!strcmp(argv[1], "run")) |
| 2062 | return maintenance_run(argc - 1, argv + 1, prefix); |
Derrick Stolee | 2fec604 | 2020-09-11 17:49:18 +0000 | [diff] [blame] | 2063 | if (!strcmp(argv[1], "start")) |
| 2064 | return maintenance_start(); |
| 2065 | if (!strcmp(argv[1], "stop")) |
| 2066 | return maintenance_stop(); |
Derrick Stolee | 0c18b70 | 2020-09-11 17:49:17 +0000 | [diff] [blame] | 2067 | if (!strcmp(argv[1], "register")) |
| 2068 | return maintenance_register(); |
| 2069 | if (!strcmp(argv[1], "unregister")) |
| 2070 | return maintenance_unregister(); |
Derrick Stolee | 2057d75 | 2020-09-17 18:11:42 +0000 | [diff] [blame] | 2071 | |
| 2072 | die(_("invalid subcommand: %s"), argv[1]); |
| 2073 | } |