add.c 16.4 KB
Newer Older
L
Linus Torvalds 已提交
1 2 3 4 5 6 7 8
/*
 * "git add" builtin command
 *
 * Copyright (C) 2006 Linus Torvalds
 */
#include "cache.h"
#include "builtin.h"
#include "dir.h"
9
#include "pathspec.h"
J
Junio C Hamano 已提交
10
#include "exec_cmd.h"
11
#include "cache-tree.h"
12
#include "run-command.h"
13
#include "parse-options.h"
14
#include "diff.h"
15
#include "diffcore.h"
16
#include "revision.h"
17
#include "bulk-checkin.h"
L
Linus Torvalds 已提交
18

19
static const char * const builtin_add_usage[] = {
20
	N_("git add [options] [--] <pathspec>..."),
21 22
	NULL
};
23
static int patch_interactive, add_interactive, edit_interactive;
J
Jeff King 已提交
24
static int take_worktree_changes;
25

26
struct update_callback_data {
27 28
	int flags;
	int add_errors;
29 30
	const char *implicit_dot;
	size_t implicit_dot_len;
31 32 33

	/* only needed for 2.0 transition preparation */
	int warn_add_would_remove;
34 35
};

36 37 38 39 40
static const char *option_with_implicit_dot;
static const char *short_option_with_implicit_dot;

static void warn_pathless_add(void)
{
41
	static int shown;
42 43
	assert(option_with_implicit_dot && short_option_with_implicit_dot);

44 45 46 47
	if (shown)
		return;
	shown = 1;

48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69
	/*
	 * To be consistent with "git add -p" and most Git
	 * commands, we should default to being tree-wide, but
	 * this is not the original behavior and can't be
	 * changed until users trained themselves not to type
	 * "git add -u" or "git add -A". For now, we warn and
	 * keep the old behavior. Later, the behavior can be changed
	 * to tree-wide, keeping the warning for a while, and
	 * eventually we can drop the warning.
	 */
	warning(_("The behavior of 'git add %s (or %s)' with no path argument from a\n"
		  "subdirectory of the tree will change in Git 2.0 and should not be used anymore.\n"
		  "To add content for the whole tree, run:\n"
		  "\n"
		  "  git add %s :/\n"
		  "  (or git add %s :/)\n"
		  "\n"
		  "To restrict the command to the current directory, run:\n"
		  "\n"
		  "  git add %s .\n"
		  "  (or git add %s .)\n"
		  "\n"
70 71 72
		  "With the current Git version, the command is restricted to "
		  "the current directory.\n"
		  ""),
73 74 75 76 77
		option_with_implicit_dot, short_option_with_implicit_dot,
		option_with_implicit_dot, short_option_with_implicit_dot,
		option_with_implicit_dot, short_option_with_implicit_dot);
}

78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98
static int fix_unmerged_status(struct diff_filepair *p,
			       struct update_callback_data *data)
{
	if (p->status != DIFF_STATUS_UNMERGED)
		return p->status;
	if (!(data->flags & ADD_CACHE_IGNORE_REMOVAL) && !p->two->mode)
		/*
		 * This is not an explicit add request, and the
		 * path is missing from the working tree (deleted)
		 */
		return DIFF_STATUS_DELETED;
	else
		/*
		 * Either an explicit add request, or path exists
		 * in the working tree.  An attempt to explicitly
		 * add a path that does not exist in the working tree
		 * will be caught as an error by the caller immediately.
		 */
		return DIFF_STATUS_MODIFIED;
}

99 100 101 102 103 104 105 106 107 108 109 110 111
static const char *add_would_remove_warning = N_(
	"You ran 'git add' with neither '-A (--all)' or '--no-all', whose\n"
"behaviour will change in Git 2.0 with respect to paths you removed from\n"
"your working tree. Paths like '%s' that are\n"
"removed are ignored with this version of Git.\n"
"\n"
"* 'git add --no-all <pathspec>', which is the current default, ignores\n"
"  paths you removed from your working tree.\n"
"\n"
"* 'git add --all <pathspec>' will let you also record the removals.\n"
"\n"
"Run 'git status' to check the paths you removed from your working tree.\n");

112 113
static void warn_add_would_remove(const char *path)
{
114
	warning(_(add_would_remove_warning), path);
115 116
}

117 118 119 120 121
static void update_callback(struct diff_queue_struct *q,
			    struct diff_options *opt, void *cbdata)
{
	int i;
	struct update_callback_data *data = cbdata;
122 123
	const char *implicit_dot = data->implicit_dot;
	size_t implicit_dot_len = data->implicit_dot_len;
124 125 126 127

	for (i = 0; i < q->nr; i++) {
		struct diff_filepair *p = q->queue[i];
		const char *path = p->one->path;
128 129 130 131 132 133 134 135 136 137 138 139 140 141
		/*
		 * Check if "git add -A" or "git add -u" was run from a
		 * subdirectory with a modified file outside that directory,
		 * and warn if so.
		 *
		 * "git add -u" will behave like "git add -u :/" instead of
		 * "git add -u ." in the future.  This warning prepares for
		 * that change.
		 */
		if (implicit_dot &&
		    strncmp_icase(path, implicit_dot, implicit_dot_len)) {
			warn_pathless_add();
			continue;
		}
142
		switch (fix_unmerged_status(p, data)) {
143
		default:
144
			die(_("unexpected diff status %c"), p->status);
145 146 147 148
		case DIFF_STATUS_MODIFIED:
		case DIFF_STATUS_TYPE_CHANGED:
			if (add_file_to_index(&the_index, path, data->flags)) {
				if (!(data->flags & ADD_CACHE_IGNORE_ERRORS))
149
					die(_("updating files failed"));
150 151 152 153
				data->add_errors++;
			}
			break;
		case DIFF_STATUS_DELETED:
154 155 156 157
			if (data->warn_add_would_remove) {
				warn_add_would_remove(path);
				data->warn_add_would_remove = 0;
			}
158 159 160 161 162
			if (data->flags & ADD_CACHE_IGNORE_REMOVAL)
				break;
			if (!(data->flags & ADD_CACHE_PRETEND))
				remove_file_from_index(&the_index, path);
			if (data->flags & (ADD_CACHE_PRETEND|ADD_CACHE_VERBOSE))
163
				printf(_("remove '%s'\n"), path);
164 165 166 167 168
			break;
		}
	}
}

169 170
static void update_files_in_cache(const char *prefix, const char **pathspec,
				  struct update_callback_data *data)
171 172
{
	struct rev_info rev;
173

174 175
	init_revisions(&rev, prefix);
	setup_revisions(0, NULL, &rev, NULL);
176
	init_pathspec(&rev.prune_data, pathspec);
177 178
	rev.diffopt.output_format = DIFF_FORMAT_CALLBACK;
	rev.diffopt.format_callback = update_callback;
179
	rev.diffopt.format_callback_data = data;
180
	rev.max_count = 0; /* do not compare unmerged paths with stage #2 */
181
	run_diff_files(&rev, DIFF_RACY_IS_MODIFIED);
182 183 184 185 186 187 188 189 190
}

int add_files_to_cache(const char *prefix, const char **pathspec, int flags)
{
	struct update_callback_data data;

	memset(&data, 0, sizeof(data));
	data.flags = flags;
	update_files_in_cache(prefix, pathspec, &data);
191 192 193
	return !!data.add_errors;
}

194 195 196
#define WARN_IMPLICIT_DOT (1u << 0)
static char *prune_directory(struct dir_struct *dir, const char **pathspec,
			     int prefix, unsigned flag)
L
Linus Torvalds 已提交
197
{
198 199
	char *seen;
	int i, specs;
L
Linus Torvalds 已提交
200 201
	struct dir_entry **src, **dst;

202 203
	for (specs = 0; pathspec[specs];  specs++)
		/* nothing */;
204
	seen = xcalloc(specs, 1);
205

L
Linus Torvalds 已提交
206 207 208 209
	src = dst = dir->entries;
	i = dir->nr;
	while (--i >= 0) {
		struct dir_entry *entry = *src++;
J
Junio C Hamano 已提交
210 211 212
		if (match_pathspec(pathspec, entry->name, entry->len,
				   prefix, seen))
			*dst++ = entry;
213 214 215 216 217 218 219 220 221 222
		else if (flag & WARN_IMPLICIT_DOT)
			/*
			 * "git add -A" was run from a subdirectory with a
			 * new file outside that directory.
			 *
			 * "git add -A" will behave like "git add -A :/"
			 * instead of "git add -A ." in the future.
			 * Warn about the coming behavior change.
			 */
			warn_pathless_add();
L
Linus Torvalds 已提交
223 224
	}
	dir->nr = dst - dir->entries;
225
	add_pathspec_matches_against_index(pathspec, seen, specs);
226
	return seen;
L
Linus Torvalds 已提交
227 228
}

229 230 231 232
/*
 * Checks the index to see whether any path in pathspec refers to
 * something inside a submodule.  If so, dies with an error message.
 */
233 234 235 236 237 238 239
static void treat_gitlinks(const char **pathspec)
{
	int i;

	if (!pathspec || !*pathspec)
		return;

240 241
	for (i = 0; pathspec[i]; i++)
		pathspec[i] = check_path_for_gitlink(pathspec[i]);
242 243
}

244 245 246 247 248 249 250 251
static void refresh(int verbose, const char **pathspec)
{
	char *seen;
	int i, specs;

	for (specs = 0; pathspec[specs];  specs++)
		/* nothing */;
	seen = xcalloc(specs, 1);
252
	refresh_index(&the_index, verbose ? REFRESH_IN_PORCELAIN : REFRESH_QUIET,
253
		      pathspec, seen, _("Unstaged changes after refreshing the index:"));
254 255
	for (i = 0; i < specs; i++) {
		if (!seen[i])
256
			die(_("pathspec '%s' did not match any files"), pathspec[i]);
257
	}
258
        free(seen);
259 260
}

261 262 263 264 265
/*
 * Normalizes argv relative to prefix, via get_pathspec(), and then
 * runs die_if_path_beyond_symlink() on each path in the normalized
 * list.
 */
266
static const char **validate_pathspec(const char **argv, const char *prefix)
267 268 269
{
	const char **pathspec = get_pathspec(prefix, argv);

270 271 272
	if (pathspec) {
		const char **p;
		for (p = pathspec; *p; p++) {
273
			die_if_path_beyond_symlink(*p, prefix);
274 275 276
		}
	}

277 278 279
	return pathspec;
}

280 281
int run_add_interactive(const char *revision, const char *patch_mode,
			const char **pathspec)
282
{
283
	int status, ac, pc = 0;
284 285
	const char **args;

286 287 288
	if (pathspec)
		while (pathspec[pc])
			pc++;
289

290
	args = xcalloc(sizeof(const char *), (pc + 5));
291 292
	ac = 0;
	args[ac++] = "add--interactive";
293 294 295 296
	if (patch_mode)
		args[ac++] = patch_mode;
	if (revision)
		args[ac++] = revision;
297
	args[ac++] = "--";
298 299 300
	if (pc) {
		memcpy(&(args[ac]), pathspec, sizeof(const char *) * pc);
		ac += pc;
301 302
	}
	args[ac] = NULL;
303 304 305 306

	status = run_command_v_opt(args, RUN_GIT_CMD);
	free(args);
	return status;
307 308
}

309
int interactive_add(int argc, const char **argv, const char *prefix, int patch)
310 311 312 313
{
	const char **pathspec = NULL;

	if (argc) {
314
		pathspec = validate_pathspec(argv, prefix);
315 316 317 318 319
		if (!pathspec)
			return -1;
	}

	return run_add_interactive(NULL,
320
				   patch ? "--patch" : NULL,
321 322 323
				   pathspec);
}

324
static int edit_patch(int argc, const char **argv, const char *prefix)
325
{
326
	char *file = git_pathdup("ADD_EDIT.patch");
327
	const char *apply_argv[] = { "apply", "--recount", "--cached",
328
		NULL, NULL };
329 330 331 332 333
	struct child_process child;
	struct rev_info rev;
	int out;
	struct stat st;

334 335
	apply_argv[3] = file;

336 337 338
	git_config(git_diff_basic_config, NULL); /* no "diff" UI options */

	if (read_cache() < 0)
339
		die (_("Could not read the index"));
340 341 342 343 344 345

	init_revisions(&rev, prefix);
	rev.diffopt.context = 7;

	argc = setup_revisions(argc, argv, &rev, NULL);
	rev.diffopt.output_format = DIFF_FORMAT_PATCH;
346
	DIFF_OPT_SET(&rev.diffopt, IGNORE_DIRTY_SUBMODULES);
347
	out = open(file, O_CREAT | O_WRONLY, 0666);
348
	if (out < 0)
349
		die (_("Could not open '%s' for writing."), file);
350
	rev.diffopt.file = xfdopen(out, "w");
351 352
	rev.diffopt.close_file = 1;
	if (run_diff_files(&rev, 0))
353
		die (_("Could not write patch"));
354 355 356 357

	launch_editor(file, NULL, NULL);

	if (stat(file, &st))
358
		die_errno(_("Could not stat '%s'"), file);
359
	if (!st.st_size)
360
		die(_("Empty patch. Aborted."));
361 362 363 364 365

	memset(&child, 0, sizeof(child));
	child.git_cmd = 1;
	child.argv = apply_argv;
	if (run_command(&child))
366
		die (_("Could not apply '%s'"), file);
367 368

	unlink(file);
369
	free(file);
370 371 372
	return 0;
}

373
static struct lock_file lock_file;
L
Linus Torvalds 已提交
374

375
static const char ignore_error[] =
376
N_("The following paths are ignored by one of your .gitignore files:\n");
377

378
static int verbose, show_only, ignored_too, refresh_only;
379 380 381 382 383
static int ignore_add_errors, intent_to_add, ignore_missing;

#define ADDREMOVE_DEFAULT 0 /* Change to 1 in Git 2.0 */
static int addremove = ADDREMOVE_DEFAULT;
static int addremove_explicit = -1; /* unspecified */
384 385

static struct option builtin_add_options[] = {
386 387
	OPT__DRY_RUN(&show_only, N_("dry run")),
	OPT__VERBOSE(&verbose, N_("be verbose")),
388
	OPT_GROUP(""),
389 390 391
	OPT_BOOL('i', "interactive", &add_interactive, N_("interactive picking")),
	OPT_BOOL('p', "patch", &patch_interactive, N_("select hunks interactively")),
	OPT_BOOL('e', "edit", &edit_interactive, N_("edit current diff and apply")),
392
	OPT__FORCE(&ignored_too, N_("allow adding otherwise ignored files")),
393 394
	OPT_BOOL('u', "update", &take_worktree_changes, N_("update tracked files")),
	OPT_BOOL('N', "intent-to-add", &intent_to_add, N_("record only the fact that the path will be added later")),
395
	OPT_BOOL('A', "all", &addremove_explicit, N_("add changes from all tracked and untracked files")),
396 397 398
	OPT_BOOL( 0 , "refresh", &refresh_only, N_("don't add, only refresh the index")),
	OPT_BOOL( 0 , "ignore-errors", &ignore_add_errors, N_("just skip files which cannot be added because of errors")),
	OPT_BOOL( 0 , "ignore-missing", &ignore_missing, N_("check if - even missing - files are ignored in dry run")),
399 400 401
	OPT_END(),
};

J
Junio C Hamano 已提交
402
static int add_config(const char *var, const char *value, void *cb)
403
{
404 405
	if (!strcmp(var, "add.ignoreerrors") ||
	    !strcmp(var, "add.ignore-errors")) {
406 407 408
		ignore_add_errors = git_config_bool(var, value);
		return 0;
	}
J
Junio C Hamano 已提交
409
	return git_default_config(var, value, cb);
410 411
}

412 413 414 415 416
static int add_files(struct dir_struct *dir, int flags)
{
	int i, exit_status = 0;

	if (dir->ignored_nr) {
417
		fprintf(stderr, _(ignore_error));
418 419
		for (i = 0; i < dir->ignored_nr; i++)
			fprintf(stderr, "%s\n", dir->ignored[i]->name);
420 421
		fprintf(stderr, _("Use -f if you really want to add them.\n"));
		die(_("no files added"));
422 423 424 425 426
	}

	for (i = 0; i < dir->nr; i++)
		if (add_file_to_cache(dir->entries[i]->name, flags)) {
			if (!ignore_add_errors)
427
				die(_("adding files failed"));
428 429 430 431 432
			exit_status = 1;
		}
	return exit_status;
}

433
int cmd_add(int argc, const char **argv, const char *prefix)
L
Linus Torvalds 已提交
434
{
435
	int exit_status = 0;
436
	int newfd;
L
Linus Torvalds 已提交
437 438
	const char **pathspec;
	struct dir_struct dir;
439
	int flags;
440 441
	int add_new_files;
	int require_pathspec;
442
	char *seen = NULL;
443
	int implicit_dot = 0;
444
	struct update_callback_data update_data;
J
Junio C Hamano 已提交
445

446 447
	git_config(add_config, NULL);

448
	argc = parse_options(argc, argv, prefix, builtin_add_options,
449
			  builtin_add_usage, PARSE_OPT_KEEP_ARGV0);
450 451
	if (patch_interactive)
		add_interactive = 1;
452
	if (add_interactive)
453
		exit(interactive_add(argc - 1, argv + 1, prefix, patch_interactive));
L
Linus Torvalds 已提交
454

455 456 457 458 459
	if (edit_interactive)
		return(edit_patch(argc, argv, prefix));
	argc--;
	argv++;

460 461 462 463 464
	if (0 <= addremove_explicit)
		addremove = addremove_explicit;
	else if (take_worktree_changes && ADDREMOVE_DEFAULT)
		addremove = 0; /* "-u" was given but not "-A" */

J
Junio C Hamano 已提交
465
	if (addremove && take_worktree_changes)
466
		die(_("-A and -u are mutually incompatible"));
467 468 469

	/*
	 * Warn when "git add pathspec..." was given without "-u" or "-A"
470
	 * and pathspec... covers a removed path.
471
	 */
472 473 474
	memset(&update_data, 0, sizeof(update_data));
	if (!take_worktree_changes && addremove_explicit < 0)
		update_data.warn_add_would_remove = 1;
475 476 477 478 479 480 481 482

	if (!take_worktree_changes && addremove_explicit < 0 && argc)
		/*
		 * Turn "git add pathspec..." to "git add -A pathspec..."
		 * in Git 2.0 but not yet
		 */
		; /* addremove = 1; */

483
	if (!show_only && ignore_missing)
484
		die(_("Option --ignore-missing can only be used together with --dry-run"));
485 486 487 488 489 490 491 492 493
	if (addremove) {
		option_with_implicit_dot = "--all";
		short_option_with_implicit_dot = "-A";
	}
	if (take_worktree_changes) {
		option_with_implicit_dot = "--update";
		short_option_with_implicit_dot = "-u";
	}
	if (option_with_implicit_dot && !argc) {
J
Junio C Hamano 已提交
494 495 496
		static const char *here[2] = { ".", NULL };
		argc = 1;
		argv = here;
497
		implicit_dot = 1;
J
Junio C Hamano 已提交
498 499
	}

500 501 502
	add_new_files = !take_worktree_changes && !refresh_only;
	require_pathspec = !take_worktree_changes;

503
	newfd = hold_locked_index(&lock_file, 1);
L
Linus Torvalds 已提交
504

505
	flags = ((verbose ? ADD_CACHE_VERBOSE : 0) |
J
Junio C Hamano 已提交
506
		 (show_only ? ADD_CACHE_PRETEND : 0) |
J
Junio C Hamano 已提交
507
		 (intent_to_add ? ADD_CACHE_INTENT : 0) |
508 509
		 (ignore_add_errors ? ADD_CACHE_IGNORE_ERRORS : 0) |
		 (!(addremove || take_worktree_changes)
510 511
		  ? ADD_CACHE_IGNORE_REMOVAL : 0)) |
		 (implicit_dot ? ADD_CACHE_IMPLICIT_DOT : 0);
512

513
	if (require_pathspec && argc == 0) {
514 515
		fprintf(stderr, _("Nothing specified, nothing added.\n"));
		fprintf(stderr, _("Maybe you wanted to say 'git add .'?\n"));
516 517
		return 0;
	}
518
	pathspec = validate_pathspec(argv, prefix);
L
Linus Torvalds 已提交
519

520
	if (read_cache() < 0)
521
		die(_("index file corrupt"));
522
	treat_gitlinks(pathspec);
523

524 525 526 527 528 529 530 531 532 533
	if (add_new_files) {
		int baselen;

		/* Set up the default git porcelain excludes */
		memset(&dir, 0, sizeof(dir));
		if (!ignored_too) {
			dir.flags |= DIR_COLLECT_IGNORED;
			setup_standard_excludes(&dir);
		}

534
		/* This picks up the paths that are not tracked */
535
		baselen = fill_directory(&dir, implicit_dot ? NULL : pathspec);
536
		if (pathspec)
537 538
			seen = prune_directory(&dir, pathspec, baselen,
					implicit_dot ? WARN_IMPLICIT_DOT : 0);
539
	}
540

541 542 543
	if (refresh_only) {
		refresh(verbose, pathspec);
		goto finish;
544
	}
545 546
	if (implicit_dot && prefix)
		refresh_cache(REFRESH_QUIET);
547

548 549
	if (pathspec) {
		int i;
550

551
		if (!seen)
552
			seen = find_pathspecs_matching_against_index(pathspec);
553 554
		for (i = 0; pathspec[i]; i++) {
			if (!seen[i] && pathspec[i][0]
555 556
			    && !file_exists(pathspec[i])) {
				if (ignore_missing) {
557
					int dtype = DT_UNKNOWN;
558
					if (is_excluded(&dir, pathspec[i], &dtype))
559 560
						dir_add_ignored(&dir, pathspec[i], strlen(pathspec[i]));
				} else
561
					die(_("pathspec '%s' did not match any files"),
562 563
					    pathspec[i]);
			}
564 565 566 567
		}
		free(seen);
	}

568 569
	plug_bulk_checkin();

570 571 572 573 574 575 576 577 578 579 580
	if ((flags & ADD_CACHE_IMPLICIT_DOT) && prefix) {
		/*
		 * Check for modified files throughout the worktree so
		 * update_callback has a chance to warn about changes
		 * outside the cwd.
		 */
		update_data.implicit_dot = prefix;
		update_data.implicit_dot_len = strlen(prefix);
		pathspec = NULL;
	}
	update_data.flags = flags & ~ADD_CACHE_IMPLICIT_DOT;
581
	update_files_in_cache(prefix, pathspec, &update_data);
582

583
	exit_status |= !!update_data.add_errors;
584 585
	if (add_new_files)
		exit_status |= add_files(&dir, flags);
L
Linus Torvalds 已提交
586

587 588
	unplug_bulk_checkin();

589
 finish:
L
Linus Torvalds 已提交
590 591
	if (active_cache_changed) {
		if (write_cache(newfd, active_cache, active_nr) ||
B
Brandon Casey 已提交
592
		    commit_locked_index(&lock_file))
593
			die(_("Unable to write new index file"));
L
Linus Torvalds 已提交
594 595
	}

596
	return exit_status;
L
Linus Torvalds 已提交
597
}