builtin-log.c 11.1 KB
Newer Older
1 2 3 4 5 6 7 8 9 10 11
/*
 * Builtin "git log" and related commands (show, whatchanged)
 *
 * (C) Copyright 2006 Linus Torvalds
 *		 2006 Junio Hamano
 */
#include "cache.h"
#include "commit.h"
#include "diff.h"
#include "revision.h"
#include "log-tree.h"
12
#include "builtin.h"
13 14
#include <time.h>
#include <sys/time.h>
15

16 17 18
/* this is in builtin-diff.c */
void add_head(struct rev_info *revs);

19
static void cmd_log_init(int argc, const char **argv, const char *prefix,
20 21 22 23 24 25
		      struct rev_info *rev)
{
	rev->abbrev = DEFAULT_ABBREV;
	rev->commit_format = CMIT_FMT_DEFAULT;
	rev->verbose_header = 1;
	argc = setup_revisions(argc, argv, rev, "HEAD");
26 27
	if (rev->diffopt.pickaxe || rev->diffopt.filter)
		rev->always_show_header = 0;
28 29
	if (argc > 1)
		die("unrecognized argument: %s", argv[1]);
30 31 32 33 34
}

static int cmd_log_walk(struct rev_info *rev)
{
	struct commit *commit;
35 36 37 38 39 40

	prepare_revision_walk(rev);
	while ((commit = get_revision(rev)) != NULL) {
		log_tree_commit(rev, commit);
		free(commit->buffer);
		commit->buffer = NULL;
L
Linus Torvalds 已提交
41 42
		free_commit_list(commit->parents);
		commit->parents = NULL;
43 44 45 46
	}
	return 0;
}

47
int cmd_whatchanged(int argc, const char **argv, const char *prefix)
48 49 50
{
	struct rev_info rev;

51
	git_config(git_diff_ui_config);
52
	init_revisions(&rev, prefix);
53 54
	rev.diff = 1;
	rev.diffopt.recursive = 1;
L
Linus Torvalds 已提交
55
	rev.simplify_history = 0;
56
	cmd_log_init(argc, argv, prefix, &rev);
57 58 59
	if (!rev.diffopt.output_format)
		rev.diffopt.output_format = DIFF_FORMAT_RAW;
	return cmd_log_walk(&rev);
60 61
}

62
int cmd_show(int argc, const char **argv, const char *prefix)
63 64 65
{
	struct rev_info rev;

66
	git_config(git_diff_ui_config);
67
	init_revisions(&rev, prefix);
68 69 70 71 72 73 74
	rev.diff = 1;
	rev.diffopt.recursive = 1;
	rev.combine_merges = 1;
	rev.dense_combined_merges = 1;
	rev.always_show_header = 1;
	rev.ignore_merges = 0;
	rev.no_walk = 1;
75
	cmd_log_init(argc, argv, prefix, &rev);
76
	return cmd_log_walk(&rev);
77 78
}

79
int cmd_log(int argc, const char **argv, const char *prefix)
80 81 82
{
	struct rev_info rev;

83
	git_config(git_diff_ui_config);
84
	init_revisions(&rev, prefix);
85
	rev.always_show_header = 1;
86
	cmd_log_init(argc, argv, prefix, &rev);
87
	return cmd_log_walk(&rev);
88
}
89

90 91 92 93 94 95
static int istitlechar(char c)
{
	return (c >= 'a' && c <= 'z') || (c >= 'A' && c <= 'Z') ||
		(c >= '0' && c <= '9') || c == '.' || c == '_';
}

96 97 98 99 100 101 102 103
static char *extra_headers = NULL;
static int extra_headers_size = 0;

static int git_format_config(const char *var, const char *value)
{
	if (!strcmp(var, "format.headers")) {
		int len = strlen(value);
		extra_headers_size += len + 1;
J
Jonas Fonseca 已提交
104
		extra_headers = xrealloc(extra_headers, extra_headers_size);
105 106 107 108
		extra_headers[extra_headers_size - len - 1] = 0;
		strcat(extra_headers, value);
		return 0;
	}
109 110 111
	if (!strcmp(var, "diff.color")) {
		return 0;
	}
112
	return git_diff_ui_config(var, value);
113 114 115
}


116
static FILE *realstdout = NULL;
117
static const char *output_directory = NULL;
118

119
static void reopen_stdout(struct commit *commit, int nr, int keep_subject)
120 121 122
{
	char filename[1024];
	char *sol;
123
	int len = 0;
124

125
	if (output_directory) {
126
		strlcpy(filename, output_directory, 1010);
127 128 129 130
		len = strlen(filename);
		if (filename[len - 1] != '/')
			filename[len++] = '/';
	}
131

132
	sprintf(filename + len, "%04d", nr);
133 134 135 136 137 138 139 140
	len = strlen(filename);

	sol = strstr(commit->buffer, "\n\n");
	if (sol) {
		int j, space = 1;

		sol += 2;
		/* strip [PATCH] or [PATCH blabla] */
141
		if (!keep_subject && !strncmp(sol, "[PATCH", 6)) {
142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166
			char *eos = strchr(sol + 6, ']');
			if (eos) {
				while (isspace(*eos))
					eos++;
				sol = eos;
			}
		}

		for (j = 0; len < 1024 - 6 && sol[j] && sol[j] != '\n'; j++) {
			if (istitlechar(sol[j])) {
				if (space) {
					filename[len++] = '-';
					space = 0;
				}
				filename[len++] = sol[j];
				if (sol[j] == '.')
					while (sol[j + 1] == '.')
						j++;
			} else
				space = 1;
		}
		while (filename[len - 1] == '.' || filename[len - 1] == '-')
			len--;
	}
	strcpy(filename + len, ".txt");
167
	fprintf(realstdout, "%s\n", filename);
168 169 170
	freopen(filename, "w", stdout);
}

171 172 173 174 175 176 177 178 179
static int get_patch_id(struct commit *commit, struct diff_options *options,
		unsigned char *sha1)
{
	diff_tree_sha1(commit->parents->item->object.sha1, commit->object.sha1,
			"", options);
	diffcore_std(options);
	return diff_flush_patch_id(options, sha1);
}

180
static void get_patch_ids(struct rev_info *rev, struct diff_options *options, const char *prefix)
181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204
{
	struct rev_info check_rev;
	struct commit *commit;
	struct object *o1, *o2;
	unsigned flags1, flags2;
	unsigned char sha1[20];

	if (rev->pending.nr != 2)
		die("Need exactly one range.");

	o1 = rev->pending.objects[0].item;
	flags1 = o1->flags;
	o2 = rev->pending.objects[1].item;
	flags2 = o2->flags;

	if ((flags1 & UNINTERESTING) == (flags2 & UNINTERESTING))
		die("Not a range.");

	diff_setup(options);
	options->recursive = 1;
	if (diff_setup_done(options) < 0)
		die("diff_setup_done failed");

	/* given a range a..b get all patch ids for b..a */
205
	init_revisions(&check_rev, prefix);
206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221
	o1->flags ^= UNINTERESTING;
	o2->flags ^= UNINTERESTING;
	add_pending_object(&check_rev, o1, "o1");
	add_pending_object(&check_rev, o2, "o2");
	prepare_revision_walk(&check_rev);

	while ((commit = get_revision(&check_rev)) != NULL) {
		/* ignore merges */
		if (commit->parents && commit->parents->next)
			continue;

		if (!get_patch_id(commit, options, sha1))
			created_object(sha1, xcalloc(1, sizeof(struct object)));
	}

	/* reset for next revision walk */
222 223 224 225
	clear_commit_marks((struct commit *)o1,
			SEEN | UNINTERESTING | SHOWN | ADDED);
	clear_commit_marks((struct commit *)o2,
			SEEN | UNINTERESTING | SHOWN | ADDED);
226 227 228 229
	o1->flags = flags1;
	o2->flags = flags2;
}

230 231 232 233 234 235 236
static void gen_message_id(char *dest, unsigned int length, char *base)
{
	const char *committer = git_committer_info(1);
	const char *email_start = strrchr(committer, '<');
	const char *email_end = strrchr(committer, '>');
	if(!email_start || !email_end || email_start > email_end - 1)
		die("Could not extract email from committer identity.");
237 238 239
	snprintf(dest, length, "%s.%lu.git.%.*s", base,
		 (unsigned long) time(NULL),
		 (int)(email_end - email_start - 1), email_start + 1);
240 241
}

242
int cmd_format_patch(int argc, const char **argv, const char *prefix)
243 244 245 246
{
	struct commit *commit;
	struct commit **list = NULL;
	struct rev_info rev;
247
	int nr = 0, total, i, j;
248
	int use_stdout = 0;
249
	int numbered = 0;
250
	int start_number = -1;
251
	int keep_subject = 0;
252
	int ignore_if_in_upstream = 0;
253
	int thread = 0;
254
	const char *in_reply_to = NULL;
255
	struct diff_options patch_id_opts;
J
Junio C Hamano 已提交
256
	char *add_signoff = NULL;
257 258
	char message_id[1024];
	char ref_message_id[1024];
259

260
	setup_ident();
261
	git_config(git_format_config);
262
	init_revisions(&rev, prefix);
263 264 265 266 267
	rev.commit_format = CMIT_FMT_EMAIL;
	rev.verbose_header = 1;
	rev.diff = 1;
	rev.combine_merges = 0;
	rev.ignore_merges = 1;
268 269
	rev.diffopt.msg_sep = "";
	rev.diffopt.recursive = 1;
270

271 272
	rev.extra_headers = extra_headers;

273 274 275 276 277 278 279
	/*
	 * Parse the arguments before setup_revisions(), or something
	 * like "git fmt-patch -o a123 HEAD^.." may fail; a123 is
	 * possibly a valid SHA1.
	 */
	for (i = 1, j = 1; i < argc; i++) {
		if (!strcmp(argv[i], "--stdout"))
280
			use_stdout = 1;
281 282 283
		else if (!strcmp(argv[i], "-n") ||
				!strcmp(argv[i], "--numbered"))
			numbered = 1;
284 285 286 287 288 289 290
		else if (!strncmp(argv[i], "--start-number=", 15))
			start_number = strtol(argv[i] + 15, NULL, 10);
		else if (!strcmp(argv[i], "--start-number")) {
			i++;
			if (i == argc)
				die("Need a number for --start-number");
			start_number = strtol(argv[i], NULL, 10);
J
Junio C Hamano 已提交
291 292
		}
		else if (!strcmp(argv[i], "-k") ||
293 294 295
				!strcmp(argv[i], "--keep-subject")) {
			keep_subject = 1;
			rev.total = -1;
J
Junio C Hamano 已提交
296
		}
297 298
		else if (!strcmp(argv[i], "--output-directory") ||
			 !strcmp(argv[i], "-o")) {
299
			i++;
300 301 302 303 304
			if (argc <= i)
				die("Which directory?");
			if (output_directory)
				die("Two output directories?");
			output_directory = argv[i];
305
		}
J
Junio C Hamano 已提交
306 307
		else if (!strcmp(argv[i], "--signoff") ||
			 !strcmp(argv[i], "-s")) {
E
Eric W. Biederman 已提交
308 309 310 311
			const char *committer;
			const char *endpos;
			committer = git_committer_info(1);
			endpos = strchr(committer, '>');
J
Junio C Hamano 已提交
312 313 314 315 316 317
			if (!endpos)
				die("bogos committer info %s\n", committer);
			add_signoff = xmalloc(endpos - committer + 2);
			memcpy(add_signoff, committer, endpos - committer + 1);
			add_signoff[endpos - committer + 1] = 0;
		}
318 319 320 321
		else if (!strcmp(argv[i], "--attach"))
			rev.mime_boundary = git_version_string;
		else if (!strncmp(argv[i], "--attach=", 9))
			rev.mime_boundary = argv[i] + 9;
322 323
		else if (!strcmp(argv[i], "--ignore-if-in-upstream"))
			ignore_if_in_upstream = 1;
324 325
		else if (!strcmp(argv[i], "--thread"))
			thread = 1;
326 327 328 329 330 331 332 333
		else if (!strncmp(argv[i], "--in-reply-to=", 14))
			in_reply_to = argv[i] + 14;
		else if (!strcmp(argv[i], "--in-reply-to")) {
			i++;
			if (i == argc)
				die("Need a Message-Id for --in-reply-to");
			in_reply_to = argv[i];
		}
334
		else
335
			argv[j++] = argv[i];
336
	}
337 338
	argc = j;

339
	if (start_number < 0)
340
		start_number = 1;
341
	if (numbered && keep_subject)
342 343
		die ("-n and -k are mutually exclusive.");

344 345 346
	argc = setup_revisions(argc, argv, &rev, "HEAD");
	if (argc > 1)
		die ("unrecognized argument: %s", argv[1]);
347

348 349 350
	if (!rev.diffopt.output_format)
		rev.diffopt.output_format = DIFF_FORMAT_DIFFSTAT | DIFF_FORMAT_PATCH;

351 352 353
	if (!output_directory)
		output_directory = prefix;

354 355 356 357 358 359 360 361
	if (output_directory) {
		if (use_stdout)
			die("standard output, or directory, which one?");
		if (mkdir(output_directory, 0777) < 0 && errno != EEXIST)
			die("Could not create directory %s",
			    output_directory);
	}

362 363
	if (rev.pending.nr == 1) {
		rev.pending.objects[0].item->flags |= UNINTERESTING;
364 365 366
		add_head(&rev);
	}

367
	if (ignore_if_in_upstream)
368
		get_patch_ids(&rev, &patch_id_opts, prefix);
369

370 371 372
	if (!use_stdout)
		realstdout = fdopen(dup(1), "w");

373 374
	prepare_revision_walk(&rev);
	while ((commit = get_revision(&rev)) != NULL) {
375 376
		unsigned char sha1[20];

377 378 379
		/* ignore merges */
		if (commit->parents && commit->parents->next)
			continue;
380 381 382 383 384 385

		if (ignore_if_in_upstream &&
				!get_patch_id(commit, &patch_id_opts, sha1) &&
				lookup_object(sha1))
			continue;

386
		nr++;
J
Jonas Fonseca 已提交
387
		list = xrealloc(list, nr * sizeof(list[0]));
388 389
		list[nr - 1] = commit;
	}
390
	total = nr;
391
	if (numbered)
392
		rev.total = total + start_number - 1;
J
Junio C Hamano 已提交
393
	rev.add_signoff = add_signoff;
394
	rev.ref_message_id = in_reply_to;
395 396 397
	while (0 <= --nr) {
		int shown;
		commit = list[nr];
398
		rev.nr = total - nr + (start_number - 1);
399
		/* Make the second and subsequent mails replies to the first */
400 401 402 403 404 405 406 407 408 409
		if (thread) {
			if (nr == (total - 2)) {
				strncpy(ref_message_id, message_id,
					sizeof(ref_message_id));
				ref_message_id[sizeof(ref_message_id)-1]='\0';
				rev.ref_message_id = ref_message_id;
			}
			gen_message_id(message_id, sizeof(message_id),
				       sha1_to_hex(commit->object.sha1));
			rev.message_id = message_id;
410
		}
411
		if (!use_stdout)
412
			reopen_stdout(commit, rev.nr, keep_subject);
413 414 415
		shown = log_tree_commit(&rev, commit);
		free(commit->buffer);
		commit->buffer = NULL;
416 417 418 419 420 421 422 423 424

		/* We put one extra blank line between formatted
		 * patches and this flag is used by log-tree code
		 * to see if it needs to emit a LF before showing
		 * the log; when using one file per patch, we do
		 * not want the extra blank line.
		 */
		if (!use_stdout)
			rev.shown_one = 0;
425 426 427 428 429 430 431 432
		if (shown) {
			if (rev.mime_boundary)
				printf("\n--%s%s--\n\n\n",
				       mime_boundary_leader,
				       rev.mime_boundary);
			else
				printf("-- \n%s\n\n", git_version_string);
		}
433 434
		if (!use_stdout)
			fclose(stdout);
435 436 437 438 439
	}
	free(list);
	return 0;
}