commit
						c99fec6e35
					
				|  | @ -0,0 +1,22 @@ | ||||||
|  | Git v2.3.8 Release Notes | ||||||
|  | ======================== | ||||||
|  |  | ||||||
|  | Fixes since v2.3.7 | ||||||
|  | ------------------ | ||||||
|  |  | ||||||
|  |  * The usual "git diff" when seeing a file turning into a directory | ||||||
|  |    showed a patchset to remove the file and create all files in the | ||||||
|  |    directory, but "git diff --no-index" simply refused to work.  Also, | ||||||
|  |    when asked to compare a file and a directory, imitate POSIX "diff" | ||||||
|  |    and compare the file with the file with the same name in the | ||||||
|  |    directory, instead of refusing to run. | ||||||
|  |  | ||||||
|  |  * The default $HOME/.gitconfig file created upon "git config --global" | ||||||
|  |    that edits it had incorrectly spelled user.name and user.email | ||||||
|  |    entries in it. | ||||||
|  |  | ||||||
|  |  * "git commit --date=now" or anything that relies on approxidate lost | ||||||
|  |    the daylight-saving-time offset. | ||||||
|  |  | ||||||
|  | Also contains typofixes, documentation updates and trivial code | ||||||
|  | clean-ups. | ||||||
|  | @ -10,7 +10,7 @@ | ||||||
| 	Include additional statistics at the end of blame output. | 	Include additional statistics at the end of blame output. | ||||||
|  |  | ||||||
| -L <start>,<end>:: | -L <start>,<end>:: | ||||||
| -L :<regex>:: | -L :<funcname>:: | ||||||
| 	Annotate only the given line range. May be specified multiple times. | 	Annotate only the given line range. May be specified multiple times. | ||||||
| 	Overlapping ranges are allowed. | 	Overlapping ranges are allowed. | ||||||
| + | + | ||||||
|  |  | ||||||
|  | @ -62,9 +62,9 @@ produced by `--stat`, etc. | ||||||
| 	output by allowing them to allocate space in advance. | 	output by allowing them to allocate space in advance. | ||||||
|  |  | ||||||
| -L <start>,<end>:<file>:: | -L <start>,<end>:<file>:: | ||||||
| -L :<regex>:<file>:: | -L :<funcname>:<file>:: | ||||||
| 	Trace the evolution of the line range given by "<start>,<end>" | 	Trace the evolution of the line range given by "<start>,<end>" | ||||||
| 	(or the funcname regex <regex>) within the <file>.  You may | 	(or the function name regex <funcname>) within the <file>.  You may | ||||||
| 	not give any pathspec limiters.  This is currently limited to | 	not give any pathspec limiters.  This is currently limited to | ||||||
| 	a walk starting from a single revision, i.e., you may only | 	a walk starting from a single revision, i.e., you may only | ||||||
| 	give zero or one positive revision arguments. | 	give zero or one positive revision arguments. | ||||||
|  |  | ||||||
|  | @ -48,9 +48,10 @@ Documentation for older releases are available here: | ||||||
| * release notes for | * release notes for | ||||||
|   link:RelNotes/2.4.0.txt[2.4]. |   link:RelNotes/2.4.0.txt[2.4]. | ||||||
|  |  | ||||||
| * link:v2.3.7/git.html[documentation for release 2.3.7] | * link:v2.3.8/git.html[documentation for release 2.3.8] | ||||||
|  |  | ||||||
| * release notes for | * release notes for | ||||||
|  |   link:RelNotes/2.3.8.txt[2.3.8], | ||||||
|   link:RelNotes/2.3.7.txt[2.3.7], |   link:RelNotes/2.3.7.txt[2.3.7], | ||||||
|   link:RelNotes/2.3.6.txt[2.3.6], |   link:RelNotes/2.3.6.txt[2.3.6], | ||||||
|   link:RelNotes/2.3.5.txt[2.3.5], |   link:RelNotes/2.3.5.txt[2.3.5], | ||||||
|  |  | ||||||
|  | @ -99,10 +99,10 @@ linkgit:git-rev-list[1] for a complete list. | ||||||
| 	detailed explanation.) | 	detailed explanation.) | ||||||
|  |  | ||||||
| -L<start>,<end>:<file>:: | -L<start>,<end>:<file>:: | ||||||
| -L:<regex>:<file>:: | -L:<funcname>:<file>:: | ||||||
|  |  | ||||||
| 	Trace the evolution of the line range given by "<start>,<end>" | 	Trace the evolution of the line range given by "<start>,<end>" | ||||||
| 	(or the funcname regex <regex>) within the <file>.  You may | 	(or the function name regex <funcname>) within the <file>.  You may | ||||||
| 	not give any pathspec limiters.  This is currently limited to | 	not give any pathspec limiters.  This is currently limited to | ||||||
| 	a walk starting from a single revision, i.e., you may only | 	a walk starting from a single revision, i.e., you may only | ||||||
| 	give zero or one positive revision arguments. | 	give zero or one positive revision arguments. | ||||||
|  |  | ||||||
|  | @ -22,8 +22,9 @@ This is only valid for <end> and will specify a number | ||||||
| of lines before or after the line given by <start>. | of lines before or after the line given by <start>. | ||||||
|  |  | ||||||
| + | + | ||||||
| If ``:<regex>'' is given in place of <start> and <end>, it denotes the range | If ``:<funcname>'' is given in place of <start> and <end>, it is a | ||||||
| from the first funcname line that matches <regex>, up to the next | regular expression that denotes the range from the first funcname line | ||||||
| funcname line. ``:<regex>'' searches from the end of the previous `-L` range, | that matches <funcname>, up to the next funcname line. ``:<funcname>'' | ||||||
| if any, otherwise from the start of file. | searches from the end of the previous `-L` range, if any, otherwise | ||||||
| ``^:<regex>'' searches from the start of file. | from the start of file. ``^:<funcname>'' searches from the start of | ||||||
|  | file. | ||||||
|  |  | ||||||
|  | @ -455,9 +455,9 @@ static char *default_user_config(void) | ||||||
| 	struct strbuf buf = STRBUF_INIT; | 	struct strbuf buf = STRBUF_INIT; | ||||||
| 	strbuf_addf(&buf, | 	strbuf_addf(&buf, | ||||||
| 		    _("# This is Git's per-user configuration file.\n" | 		    _("# This is Git's per-user configuration file.\n" | ||||||
| 		      "[core]\n" | 		      "[user]\n" | ||||||
| 		      "# Please adapt and uncomment the following lines:\n" | 		      "# Please adapt and uncomment the following lines:\n" | ||||||
| 		      "#	user = %s\n" | 		      "#	name = %s\n" | ||||||
| 		      "#	email = %s\n"), | 		      "#	email = %s\n"), | ||||||
| 		    ident_default_name(), | 		    ident_default_name(), | ||||||
| 		    ident_default_email()); | 		    ident_default_email()); | ||||||
|  |  | ||||||
							
								
								
									
										14
									
								
								date.c
								
								
								
								
							
							
						
						
									
										14
									
								
								date.c
								
								
								
								
							|  | @ -704,10 +704,17 @@ int parse_date_basic(const char *date, unsigned long *timestamp, int *offset) | ||||||
| 		date += match; | 		date += match; | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	/* mktime uses local timezone */ | 	/* do not use mktime(), which uses local timezone, here */ | ||||||
| 	*timestamp = tm_to_time_t(&tm); | 	*timestamp = tm_to_time_t(&tm); | ||||||
|  | 	if (*timestamp == -1) | ||||||
|  | 		return -1; | ||||||
|  |  | ||||||
| 	if (*offset == -1) { | 	if (*offset == -1) { | ||||||
| 		time_t temp_time = mktime(&tm); | 		time_t temp_time; | ||||||
|  |  | ||||||
|  | 		/* gmtime_r() in match_digit() may have clobbered it */ | ||||||
|  | 		tm.tm_isdst = -1; | ||||||
|  | 		temp_time = mktime(&tm); | ||||||
| 		if ((time_t)*timestamp > temp_time) { | 		if ((time_t)*timestamp > temp_time) { | ||||||
| 			*offset = ((time_t)*timestamp - temp_time) / 60; | 			*offset = ((time_t)*timestamp - temp_time) / 60; | ||||||
| 		} else { | 		} else { | ||||||
|  | @ -715,9 +722,6 @@ int parse_date_basic(const char *date, unsigned long *timestamp, int *offset) | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	if (*timestamp == -1) |  | ||||||
| 		return -1; |  | ||||||
|  |  | ||||||
| 	if (!tm_gmt) | 	if (!tm_gmt) | ||||||
| 		*timestamp -= *offset * 60; | 		*timestamp -= *offset * 60; | ||||||
| 	return 0; /* success */ | 	return 0; /* success */ | ||||||
|  |  | ||||||
|  | @ -97,8 +97,27 @@ static int queue_diff(struct diff_options *o, | ||||||
| 	if (get_mode(name1, &mode1) || get_mode(name2, &mode2)) | 	if (get_mode(name1, &mode1) || get_mode(name2, &mode2)) | ||||||
| 		return -1; | 		return -1; | ||||||
|  |  | ||||||
| 	if (mode1 && mode2 && S_ISDIR(mode1) != S_ISDIR(mode2)) | 	if (mode1 && mode2 && S_ISDIR(mode1) != S_ISDIR(mode2)) { | ||||||
| 		return error("file/directory conflict: %s, %s", name1, name2); | 		struct diff_filespec *d1, *d2; | ||||||
|  |  | ||||||
|  | 		if (S_ISDIR(mode1)) { | ||||||
|  | 			/* 2 is file that is created */ | ||||||
|  | 			d1 = noindex_filespec(NULL, 0); | ||||||
|  | 			d2 = noindex_filespec(name2, mode2); | ||||||
|  | 			name2 = NULL; | ||||||
|  | 			mode2 = 0; | ||||||
|  | 		} else { | ||||||
|  | 			/* 1 is file that is deleted */ | ||||||
|  | 			d1 = noindex_filespec(name1, mode1); | ||||||
|  | 			d2 = noindex_filespec(NULL, 0); | ||||||
|  | 			name1 = NULL; | ||||||
|  | 			mode1 = 0; | ||||||
|  | 		} | ||||||
|  | 		/* emit that file */ | ||||||
|  | 		diff_queue(&diff_queued_diff, d1, d2); | ||||||
|  |  | ||||||
|  | 		/* and then let the entire directory be created or deleted */ | ||||||
|  | 	} | ||||||
|  |  | ||||||
| 	if (S_ISDIR(mode1) || S_ISDIR(mode2)) { | 	if (S_ISDIR(mode1) || S_ISDIR(mode2)) { | ||||||
| 		struct strbuf buffer1 = STRBUF_INIT; | 		struct strbuf buffer1 = STRBUF_INIT; | ||||||
|  | @ -182,12 +201,50 @@ static int queue_diff(struct diff_options *o, | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
|  |  | ||||||
|  | /* append basename of F to D */ | ||||||
|  | static void append_basename(struct strbuf *path, const char *dir, const char *file) | ||||||
|  | { | ||||||
|  | 	const char *tail = strrchr(file, '/'); | ||||||
|  |  | ||||||
|  | 	strbuf_addstr(path, dir); | ||||||
|  | 	while (path->len && path->buf[path->len - 1] == '/') | ||||||
|  | 		path->len--; | ||||||
|  | 	strbuf_addch(path, '/'); | ||||||
|  | 	strbuf_addstr(path, tail ? tail + 1 : file); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* | ||||||
|  |  * DWIM "diff D F" into "diff D/F F" and "diff F D" into "diff F D/F" | ||||||
|  |  * Note that we append the basename of F to D/, so "diff a/b/file D" | ||||||
|  |  * becomes "diff a/b/file D/file", not "diff a/b/file D/a/b/file". | ||||||
|  |  */ | ||||||
|  | static void fixup_paths(const char **path, struct strbuf *replacement) | ||||||
|  | { | ||||||
|  | 	unsigned int isdir0, isdir1; | ||||||
|  |  | ||||||
|  | 	if (path[0] == file_from_standard_input || | ||||||
|  | 	    path[1] == file_from_standard_input) | ||||||
|  | 		return; | ||||||
|  | 	isdir0 = is_directory(path[0]); | ||||||
|  | 	isdir1 = is_directory(path[1]); | ||||||
|  | 	if (isdir0 == isdir1) | ||||||
|  | 		return; | ||||||
|  | 	if (isdir0) { | ||||||
|  | 		append_basename(replacement, path[0], path[1]); | ||||||
|  | 		path[0] = replacement->buf; | ||||||
|  | 	} else { | ||||||
|  | 		append_basename(replacement, path[1], path[0]); | ||||||
|  | 		path[1] = replacement->buf; | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
| void diff_no_index(struct rev_info *revs, | void diff_no_index(struct rev_info *revs, | ||||||
| 		   int argc, const char **argv, | 		   int argc, const char **argv, | ||||||
| 		   const char *prefix) | 		   const char *prefix) | ||||||
| { | { | ||||||
| 	int i, prefixlen; | 	int i, prefixlen; | ||||||
| 	const char *paths[2]; | 	const char *paths[2]; | ||||||
|  | 	struct strbuf replacement = STRBUF_INIT; | ||||||
|  |  | ||||||
| 	diff_setup(&revs->diffopt); | 	diff_setup(&revs->diffopt); | ||||||
| 	for (i = 1; i < argc - 2; ) { | 	for (i = 1; i < argc - 2; ) { | ||||||
|  | @ -217,6 +274,9 @@ void diff_no_index(struct rev_info *revs, | ||||||
| 			p = xstrdup(prefix_filename(prefix, prefixlen, p)); | 			p = xstrdup(prefix_filename(prefix, prefixlen, p)); | ||||||
| 		paths[i] = p; | 		paths[i] = p; | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
|  | 	fixup_paths(paths, &replacement); | ||||||
|  |  | ||||||
| 	revs->diffopt.skip_stat_unmatch = 1; | 	revs->diffopt.skip_stat_unmatch = 1; | ||||||
| 	if (!revs->diffopt.output_format) | 	if (!revs->diffopt.output_format) | ||||||
| 		revs->diffopt.output_format = DIFF_FORMAT_PATCH; | 		revs->diffopt.output_format = DIFF_FORMAT_PATCH; | ||||||
|  | @ -235,6 +295,8 @@ void diff_no_index(struct rev_info *revs, | ||||||
| 	diffcore_std(&revs->diffopt); | 	diffcore_std(&revs->diffopt); | ||||||
| 	diff_flush(&revs->diffopt); | 	diff_flush(&revs->diffopt); | ||||||
|  |  | ||||||
|  | 	strbuf_release(&replacement); | ||||||
|  |  | ||||||
| 	/* | 	/* | ||||||
| 	 * The return code for --no-index imitates diff(1): | 	 * The return code for --no-index imitates diff(1): | ||||||
| 	 * 0 = no changes, 1 = changes, else error | 	 * 0 = no changes, 1 = changes, else error | ||||||
|  |  | ||||||
|  | @ -575,7 +575,7 @@ parse_lines(struct commit *commit, const char *prefix, struct string_list *args) | ||||||
|  |  | ||||||
| 		name_part = skip_range_arg(item->string); | 		name_part = skip_range_arg(item->string); | ||||||
| 		if (!name_part || *name_part != ':' || !name_part[1]) | 		if (!name_part || *name_part != ':' || !name_part[1]) | ||||||
| 			die("-L argument '%s' not of the form start,end:file", | 			die("-L argument not 'start,end:file' or ':funcname:file': %s", | ||||||
| 			    item->string); | 			    item->string); | ||||||
| 		range_part = xstrndup(item->string, name_part - item->string); | 		range_part = xstrndup(item->string, name_part - item->string); | ||||||
| 		name_part++; | 		name_part++; | ||||||
|  |  | ||||||
|  | @ -55,4 +55,38 @@ test_expect_success 'git diff --no-index executed outside repo gives correct err | ||||||
| 	) | 	) | ||||||
| ' | ' | ||||||
|  |  | ||||||
|  | test_expect_success 'diff D F and diff F D' ' | ||||||
|  | 	( | ||||||
|  | 		cd repo && | ||||||
|  | 		echo in-repo >a && | ||||||
|  | 		echo non-repo >../non/git/a && | ||||||
|  | 		mkdir sub && | ||||||
|  | 		echo sub-repo >sub/a && | ||||||
|  |  | ||||||
|  | 		test_must_fail git diff --no-index sub/a ../non/git/a >expect && | ||||||
|  | 		test_must_fail git diff --no-index sub/a ../non/git/ >actual && | ||||||
|  | 		test_cmp expect actual && | ||||||
|  |  | ||||||
|  | 		test_must_fail git diff --no-index a ../non/git/a >expect && | ||||||
|  | 		test_must_fail git diff --no-index a ../non/git/ >actual && | ||||||
|  | 		test_cmp expect actual && | ||||||
|  |  | ||||||
|  | 		test_must_fail git diff --no-index ../non/git/a a >expect && | ||||||
|  | 		test_must_fail git diff --no-index ../non/git a >actual && | ||||||
|  | 		test_cmp expect actual | ||||||
|  | 	) | ||||||
|  | ' | ||||||
|  |  | ||||||
|  | test_expect_success 'turning a file into a directory' ' | ||||||
|  | 	( | ||||||
|  | 		cd non/git && | ||||||
|  | 		mkdir d e e/sub && | ||||||
|  | 		echo 1 >d/sub && | ||||||
|  | 		echo 2 >e/sub/file && | ||||||
|  | 		printf "D\td/sub\nA\te/sub/file\n" >expect && | ||||||
|  | 		test_must_fail git diff --no-index --name-status d e >actual && | ||||||
|  | 		test_cmp expect actual | ||||||
|  | 	) | ||||||
|  | ' | ||||||
|  |  | ||||||
| test_done | test_done | ||||||
|  |  | ||||||
|  | @ -54,14 +54,14 @@ canned_test "-L 4:a.c -L 8,12:a.c simple" multiple-superset | ||||||
| canned_test "-L 8,12:a.c -L 4:a.c simple" multiple-superset | canned_test "-L 8,12:a.c -L 4:a.c simple" multiple-superset | ||||||
|  |  | ||||||
| test_bad_opts "-L" "switch.*requires a value" | test_bad_opts "-L" "switch.*requires a value" | ||||||
| test_bad_opts "-L b.c" "argument.*not of the form" | test_bad_opts "-L b.c" "argument not .start,end:file" | ||||||
| test_bad_opts "-L 1:" "argument.*not of the form" | test_bad_opts "-L 1:" "argument not .start,end:file" | ||||||
| test_bad_opts "-L 1:nonexistent" "There is no path" | test_bad_opts "-L 1:nonexistent" "There is no path" | ||||||
| test_bad_opts "-L 1:simple" "There is no path" | test_bad_opts "-L 1:simple" "There is no path" | ||||||
| test_bad_opts "-L '/foo:b.c'" "argument.*not of the form" | test_bad_opts "-L '/foo:b.c'" "argument not .start,end:file" | ||||||
| test_bad_opts "-L 1000:b.c" "has only.*lines" | test_bad_opts "-L 1000:b.c" "has only.*lines" | ||||||
| test_bad_opts "-L 1,1000:b.c" "has only.*lines" | test_bad_opts "-L 1,1000:b.c" "has only.*lines" | ||||||
| test_bad_opts "-L :b.c" "argument.*not of the form" | test_bad_opts "-L :b.c" "argument not .start,end:file" | ||||||
| test_bad_opts "-L :foo:b.c" "no match" | test_bad_opts "-L :foo:b.c" "no match" | ||||||
|  |  | ||||||
| test_expect_success '-L X (X == nlines)' ' | test_expect_success '-L X (X == nlines)' ' | ||||||
|  |  | ||||||
		Loading…
	
		Reference in New Issue
	
	 Junio C Hamano
						Junio C Hamano