Junio C Hamano | 0569e9b | 2008-05-23 22:28:56 -0700 | [diff] [blame] | 1 | /* |
| 2 | * "diff --no-index" support |
| 3 | * Copyright (c) 2007 by Johannes Schindelin |
| 4 | * Copyright (c) 2008 by Junio C Hamano |
| 5 | */ |
| 6 | |
| 7 | #include "cache.h" |
| 8 | #include "color.h" |
| 9 | #include "commit.h" |
| 10 | #include "blob.h" |
| 11 | #include "tag.h" |
| 12 | #include "diff.h" |
| 13 | #include "diffcore.h" |
| 14 | #include "revision.h" |
| 15 | #include "log-tree.h" |
| 16 | #include "builtin.h" |
Nguyễn Thái Ngọc Duy | 16bb3d7 | 2019-03-24 15:20:13 +0700 | [diff] [blame] | 17 | #include "parse-options.h" |
Johannes Schindelin | c455c87 | 2008-07-21 19:03:49 +0100 | [diff] [blame] | 18 | #include "string-list.h" |
Brian Bourn | fd3aeea | 2014-03-19 11:58:22 -0400 | [diff] [blame] | 19 | #include "dir.h" |
Junio C Hamano | 0569e9b | 2008-05-23 22:28:56 -0700 | [diff] [blame] | 20 | |
Brian Bourn | 9daf0ef | 2014-03-19 11:58:21 -0400 | [diff] [blame] | 21 | static int read_directory_contents(const char *path, struct string_list *list) |
Junio C Hamano | 0569e9b | 2008-05-23 22:28:56 -0700 | [diff] [blame] | 22 | { |
| 23 | DIR *dir; |
| 24 | struct dirent *e; |
| 25 | |
| 26 | if (!(dir = opendir(path))) |
| 27 | return error("Could not open directory %s", path); |
| 28 | |
| 29 | while ((e = readdir(dir))) |
Brian Bourn | fd3aeea | 2014-03-19 11:58:22 -0400 | [diff] [blame] | 30 | if (!is_dot_or_dotdot(e->d_name)) |
Julian Phillips | 78a395d | 2010-06-26 00:41:35 +0100 | [diff] [blame] | 31 | string_list_insert(list, e->d_name); |
Junio C Hamano | 0569e9b | 2008-05-23 22:28:56 -0700 | [diff] [blame] | 32 | |
| 33 | closedir(dir); |
| 34 | return 0; |
| 35 | } |
| 36 | |
Junio C Hamano | 4682d85 | 2012-06-27 20:14:47 -0700 | [diff] [blame] | 37 | /* |
| 38 | * This should be "(standard input)" or something, but it will |
| 39 | * probably expose many more breakages in the way no-index code |
| 40 | * is bolted onto the diff callchain. |
| 41 | */ |
| 42 | static const char file_from_standard_input[] = "-"; |
| 43 | |
Junio C Hamano | 0569e9b | 2008-05-23 22:28:56 -0700 | [diff] [blame] | 44 | static int get_mode(const char *path, int *mode) |
| 45 | { |
| 46 | struct stat st; |
| 47 | |
| 48 | if (!path || !strcmp(path, "/dev/null")) |
| 49 | *mode = 0; |
Jonathan Nieder | 380395d | 2013-05-02 20:26:08 +0100 | [diff] [blame] | 50 | #ifdef GIT_WINDOWS_NATIVE |
Johannes Schindelin | 36adb4a | 2009-03-07 16:51:33 +0100 | [diff] [blame] | 51 | else if (!strcasecmp(path, "nul")) |
| 52 | *mode = 0; |
| 53 | #endif |
Junio C Hamano | 4682d85 | 2012-06-27 20:14:47 -0700 | [diff] [blame] | 54 | else if (path == file_from_standard_input) |
Junio C Hamano | 0569e9b | 2008-05-23 22:28:56 -0700 | [diff] [blame] | 55 | *mode = create_ce_mode(0666); |
Johannes Schindelin | 418566b | 2009-01-29 17:30:51 +0100 | [diff] [blame] | 56 | else if (lstat(path, &st)) |
Junio C Hamano | 0569e9b | 2008-05-23 22:28:56 -0700 | [diff] [blame] | 57 | return error("Could not access '%s'", path); |
| 58 | else |
| 59 | *mode = st.st_mode; |
| 60 | return 0; |
| 61 | } |
| 62 | |
Junio C Hamano | 4682d85 | 2012-06-27 20:14:47 -0700 | [diff] [blame] | 63 | static int populate_from_stdin(struct diff_filespec *s) |
| 64 | { |
| 65 | struct strbuf buf = STRBUF_INIT; |
| 66 | size_t size = 0; |
| 67 | |
| 68 | if (strbuf_read(&buf, 0, 0) < 0) |
Nguyễn Thái Ngọc Duy | a1f06be | 2016-05-08 16:47:42 +0700 | [diff] [blame] | 69 | return error_errno("error while reading from stdin"); |
Junio C Hamano | 4682d85 | 2012-06-27 20:14:47 -0700 | [diff] [blame] | 70 | |
| 71 | s->should_munmap = 0; |
| 72 | s->data = strbuf_detach(&buf, &size); |
| 73 | s->size = size; |
| 74 | s->should_free = 1; |
| 75 | s->is_stdin = 1; |
| 76 | return 0; |
| 77 | } |
| 78 | |
| 79 | static struct diff_filespec *noindex_filespec(const char *name, int mode) |
| 80 | { |
| 81 | struct diff_filespec *s; |
| 82 | |
| 83 | if (!name) |
| 84 | name = "/dev/null"; |
| 85 | s = alloc_filespec(name); |
Brandon Williams | f9704c2 | 2017-05-30 10:30:50 -0700 | [diff] [blame] | 86 | fill_filespec(s, &null_oid, 0, mode); |
Junio C Hamano | 4682d85 | 2012-06-27 20:14:47 -0700 | [diff] [blame] | 87 | if (name == file_from_standard_input) |
| 88 | populate_from_stdin(s); |
| 89 | return s; |
| 90 | } |
| 91 | |
Junio C Hamano | 0569e9b | 2008-05-23 22:28:56 -0700 | [diff] [blame] | 92 | static int queue_diff(struct diff_options *o, |
Junio C Hamano | 875b91b | 2012-04-25 12:37:38 -0700 | [diff] [blame] | 93 | const char *name1, const char *name2) |
Junio C Hamano | 0569e9b | 2008-05-23 22:28:56 -0700 | [diff] [blame] | 94 | { |
| 95 | int mode1 = 0, mode2 = 0; |
| 96 | |
| 97 | if (get_mode(name1, &mode1) || get_mode(name2, &mode2)) |
| 98 | return -1; |
| 99 | |
Junio C Hamano | 0615173 | 2015-03-21 22:11:27 -0700 | [diff] [blame] | 100 | if (mode1 && mode2 && S_ISDIR(mode1) != S_ISDIR(mode2)) { |
| 101 | struct diff_filespec *d1, *d2; |
| 102 | |
| 103 | if (S_ISDIR(mode1)) { |
| 104 | /* 2 is file that is created */ |
| 105 | d1 = noindex_filespec(NULL, 0); |
| 106 | d2 = noindex_filespec(name2, mode2); |
| 107 | name2 = NULL; |
| 108 | mode2 = 0; |
| 109 | } else { |
| 110 | /* 1 is file that is deleted */ |
| 111 | d1 = noindex_filespec(name1, mode1); |
| 112 | d2 = noindex_filespec(NULL, 0); |
| 113 | name1 = NULL; |
| 114 | mode1 = 0; |
| 115 | } |
| 116 | /* emit that file */ |
| 117 | diff_queue(&diff_queued_diff, d1, d2); |
| 118 | |
| 119 | /* and then let the entire directory be created or deleted */ |
| 120 | } |
Junio C Hamano | 0569e9b | 2008-05-23 22:28:56 -0700 | [diff] [blame] | 121 | |
| 122 | if (S_ISDIR(mode1) || S_ISDIR(mode2)) { |
Junio C Hamano | 875b91b | 2012-04-25 12:37:38 -0700 | [diff] [blame] | 123 | struct strbuf buffer1 = STRBUF_INIT; |
| 124 | struct strbuf buffer2 = STRBUF_INIT; |
Thiago Farina | 183113a | 2010-07-04 16:46:19 -0300 | [diff] [blame] | 125 | struct string_list p1 = STRING_LIST_INIT_DUP; |
| 126 | struct string_list p2 = STRING_LIST_INIT_DUP; |
Junio C Hamano | 875b91b | 2012-04-25 12:37:38 -0700 | [diff] [blame] | 127 | int i1, i2, ret = 0; |
Bobby Powers | f3999e0 | 2012-05-16 10:28:31 -0400 | [diff] [blame] | 128 | size_t len1 = 0, len2 = 0; |
Junio C Hamano | 0569e9b | 2008-05-23 22:28:56 -0700 | [diff] [blame] | 129 | |
Brian Bourn | 9daf0ef | 2014-03-19 11:58:21 -0400 | [diff] [blame] | 130 | if (name1 && read_directory_contents(name1, &p1)) |
Junio C Hamano | 0569e9b | 2008-05-23 22:28:56 -0700 | [diff] [blame] | 131 | return -1; |
Brian Bourn | 9daf0ef | 2014-03-19 11:58:21 -0400 | [diff] [blame] | 132 | if (name2 && read_directory_contents(name2, &p2)) { |
Johannes Schindelin | c455c87 | 2008-07-21 19:03:49 +0100 | [diff] [blame] | 133 | string_list_clear(&p1, 0); |
Junio C Hamano | 0569e9b | 2008-05-23 22:28:56 -0700 | [diff] [blame] | 134 | return -1; |
| 135 | } |
| 136 | |
| 137 | if (name1) { |
Junio C Hamano | 875b91b | 2012-04-25 12:37:38 -0700 | [diff] [blame] | 138 | strbuf_addstr(&buffer1, name1); |
Jeff King | 00b6c17 | 2015-09-24 17:08:35 -0400 | [diff] [blame] | 139 | strbuf_complete(&buffer1, '/'); |
Bobby Powers | f3999e0 | 2012-05-16 10:28:31 -0400 | [diff] [blame] | 140 | len1 = buffer1.len; |
Junio C Hamano | 0569e9b | 2008-05-23 22:28:56 -0700 | [diff] [blame] | 141 | } |
| 142 | |
| 143 | if (name2) { |
Junio C Hamano | 875b91b | 2012-04-25 12:37:38 -0700 | [diff] [blame] | 144 | strbuf_addstr(&buffer2, name2); |
Jeff King | 00b6c17 | 2015-09-24 17:08:35 -0400 | [diff] [blame] | 145 | strbuf_complete(&buffer2, '/'); |
Bobby Powers | f3999e0 | 2012-05-16 10:28:31 -0400 | [diff] [blame] | 146 | len2 = buffer2.len; |
Junio C Hamano | 0569e9b | 2008-05-23 22:28:56 -0700 | [diff] [blame] | 147 | } |
| 148 | |
| 149 | for (i1 = i2 = 0; !ret && (i1 < p1.nr || i2 < p2.nr); ) { |
| 150 | const char *n1, *n2; |
| 151 | int comp; |
| 152 | |
Bobby Powers | f3999e0 | 2012-05-16 10:28:31 -0400 | [diff] [blame] | 153 | strbuf_setlen(&buffer1, len1); |
| 154 | strbuf_setlen(&buffer2, len2); |
| 155 | |
Junio C Hamano | 0569e9b | 2008-05-23 22:28:56 -0700 | [diff] [blame] | 156 | if (i1 == p1.nr) |
| 157 | comp = 1; |
| 158 | else if (i2 == p2.nr) |
| 159 | comp = -1; |
| 160 | else |
Junio C Hamano | 875b91b | 2012-04-25 12:37:38 -0700 | [diff] [blame] | 161 | comp = strcmp(p1.items[i1].string, p2.items[i2].string); |
Junio C Hamano | 0569e9b | 2008-05-23 22:28:56 -0700 | [diff] [blame] | 162 | |
| 163 | if (comp > 0) |
| 164 | n1 = NULL; |
| 165 | else { |
Junio C Hamano | 875b91b | 2012-04-25 12:37:38 -0700 | [diff] [blame] | 166 | strbuf_addstr(&buffer1, p1.items[i1++].string); |
| 167 | n1 = buffer1.buf; |
Junio C Hamano | 0569e9b | 2008-05-23 22:28:56 -0700 | [diff] [blame] | 168 | } |
| 169 | |
| 170 | if (comp < 0) |
| 171 | n2 = NULL; |
| 172 | else { |
Junio C Hamano | 875b91b | 2012-04-25 12:37:38 -0700 | [diff] [blame] | 173 | strbuf_addstr(&buffer2, p2.items[i2++].string); |
| 174 | n2 = buffer2.buf; |
Junio C Hamano | 0569e9b | 2008-05-23 22:28:56 -0700 | [diff] [blame] | 175 | } |
| 176 | |
| 177 | ret = queue_diff(o, n1, n2); |
| 178 | } |
Johannes Schindelin | c455c87 | 2008-07-21 19:03:49 +0100 | [diff] [blame] | 179 | string_list_clear(&p1, 0); |
| 180 | string_list_clear(&p2, 0); |
Bobby Powers | 176a335 | 2012-05-16 10:50:31 -0400 | [diff] [blame] | 181 | strbuf_release(&buffer1); |
| 182 | strbuf_release(&buffer2); |
Junio C Hamano | 0569e9b | 2008-05-23 22:28:56 -0700 | [diff] [blame] | 183 | |
| 184 | return ret; |
| 185 | } else { |
| 186 | struct diff_filespec *d1, *d2; |
| 187 | |
Brandon Williams | 0d1e0e7 | 2017-10-31 11:19:11 -0700 | [diff] [blame] | 188 | if (o->flags.reverse_diff) { |
René Scharfe | 402bf8e | 2017-01-28 22:41:47 +0100 | [diff] [blame] | 189 | SWAP(mode1, mode2); |
René Scharfe | 35d803b | 2017-01-28 22:40:58 +0100 | [diff] [blame] | 190 | SWAP(name1, name2); |
Junio C Hamano | 0569e9b | 2008-05-23 22:28:56 -0700 | [diff] [blame] | 191 | } |
| 192 | |
Junio C Hamano | 4682d85 | 2012-06-27 20:14:47 -0700 | [diff] [blame] | 193 | d1 = noindex_filespec(name1, mode1); |
| 194 | d2 = noindex_filespec(name2, mode2); |
Junio C Hamano | 0569e9b | 2008-05-23 22:28:56 -0700 | [diff] [blame] | 195 | diff_queue(&diff_queued_diff, d1, d2); |
| 196 | return 0; |
| 197 | } |
| 198 | } |
| 199 | |
Junio C Hamano | c9e1f2c | 2015-03-25 16:11:39 -0700 | [diff] [blame] | 200 | /* append basename of F to D */ |
| 201 | static void append_basename(struct strbuf *path, const char *dir, const char *file) |
| 202 | { |
| 203 | const char *tail = strrchr(file, '/'); |
| 204 | |
| 205 | strbuf_addstr(path, dir); |
| 206 | while (path->len && path->buf[path->len - 1] == '/') |
| 207 | path->len--; |
| 208 | strbuf_addch(path, '/'); |
| 209 | strbuf_addstr(path, tail ? tail + 1 : file); |
| 210 | } |
| 211 | |
| 212 | /* |
| 213 | * DWIM "diff D F" into "diff D/F F" and "diff F D" into "diff F D/F" |
| 214 | * Note that we append the basename of F to D/, so "diff a/b/file D" |
| 215 | * becomes "diff a/b/file D/file", not "diff a/b/file D/a/b/file". |
| 216 | */ |
| 217 | static void fixup_paths(const char **path, struct strbuf *replacement) |
| 218 | { |
| 219 | unsigned int isdir0, isdir1; |
| 220 | |
| 221 | if (path[0] == file_from_standard_input || |
| 222 | path[1] == file_from_standard_input) |
| 223 | return; |
| 224 | isdir0 = is_directory(path[0]); |
| 225 | isdir1 = is_directory(path[1]); |
| 226 | if (isdir0 == isdir1) |
| 227 | return; |
| 228 | if (isdir0) { |
| 229 | append_basename(replacement, path[0], path[1]); |
| 230 | path[0] = replacement->buf; |
| 231 | } else { |
| 232 | append_basename(replacement, path[1], path[0]); |
| 233 | path[1] = replacement->buf; |
| 234 | } |
| 235 | } |
| 236 | |
Nguyễn Thái Ngọc Duy | 16bb3d7 | 2019-03-24 15:20:13 +0700 | [diff] [blame] | 237 | static const char * const diff_no_index_usage[] = { |
| 238 | N_("git diff --no-index [<options>] <path> <path>"), |
| 239 | NULL |
| 240 | }; |
| 241 | |
Junio C Hamano | dcd6a8c | 2019-04-25 16:41:12 +0900 | [diff] [blame] | 242 | int diff_no_index(struct rev_info *revs, |
Nguyễn Thái Ngọc Duy | 16bb3d7 | 2019-03-24 15:20:13 +0700 | [diff] [blame] | 243 | int implicit_no_index, |
| 244 | int argc, const char **argv) |
Junio C Hamano | 0569e9b | 2008-05-23 22:28:56 -0700 | [diff] [blame] | 245 | { |
Nguyễn Thái Ngọc Duy | 16bb3d7 | 2019-03-24 15:20:13 +0700 | [diff] [blame] | 246 | int i, no_index; |
Junio C Hamano | c20f592 | 2012-06-27 11:51:15 -0700 | [diff] [blame] | 247 | const char *paths[2]; |
Junio C Hamano | c9e1f2c | 2015-03-25 16:11:39 -0700 | [diff] [blame] | 248 | struct strbuf replacement = STRBUF_INIT; |
Nguyễn Thái Ngọc Duy | e5f7a5d | 2016-01-20 18:06:02 +0700 | [diff] [blame] | 249 | const char *prefix = revs->prefix; |
Nguyễn Thái Ngọc Duy | 16bb3d7 | 2019-03-24 15:20:13 +0700 | [diff] [blame] | 250 | struct option no_index_options[] = { |
| 251 | OPT_BOOL_F(0, "no-index", &no_index, "", |
| 252 | PARSE_OPT_NONEG | PARSE_OPT_HIDDEN), |
| 253 | OPT_END(), |
| 254 | }; |
| 255 | struct option *options; |
Junio C Hamano | 0569e9b | 2008-05-23 22:28:56 -0700 | [diff] [blame] | 256 | |
Nguyễn Thái Ngọc Duy | 16bb3d7 | 2019-03-24 15:20:13 +0700 | [diff] [blame] | 257 | options = parse_options_concat(no_index_options, |
| 258 | revs->diffopt.parseopts); |
| 259 | argc = parse_options(argc, argv, revs->prefix, options, |
| 260 | diff_no_index_usage, 0); |
| 261 | if (argc != 2) { |
| 262 | if (implicit_no_index) |
| 263 | warning(_("Not a git repository. Use --no-index to " |
| 264 | "compare two paths outside a working tree")); |
| 265 | usage_with_options(diff_no_index_usage, options); |
Junio C Hamano | 0569e9b | 2008-05-23 22:28:56 -0700 | [diff] [blame] | 266 | } |
Nguyễn Thái Ngọc Duy | 16bb3d7 | 2019-03-24 15:20:13 +0700 | [diff] [blame] | 267 | FREE_AND_NULL(options); |
Junio C Hamano | 3b069b1 | 2012-06-27 12:05:52 -0700 | [diff] [blame] | 268 | for (i = 0; i < 2; i++) { |
| 269 | const char *p = argv[argc - 2 + i]; |
| 270 | if (!strcmp(p, "-")) |
Junio C Hamano | 0569e9b | 2008-05-23 22:28:56 -0700 | [diff] [blame] | 271 | /* |
Junio C Hamano | 3b069b1 | 2012-06-27 12:05:52 -0700 | [diff] [blame] | 272 | * stdin should be spelled as "-"; if you have |
| 273 | * path that is "-", spell it as "./-". |
Junio C Hamano | 0569e9b | 2008-05-23 22:28:56 -0700 | [diff] [blame] | 274 | */ |
Junio C Hamano | 4682d85 | 2012-06-27 20:14:47 -0700 | [diff] [blame] | 275 | p = file_from_standard_input; |
Jeff King | 116fb64 | 2017-03-20 21:22:28 -0400 | [diff] [blame] | 276 | else if (prefix) |
Jeff King | e4da43b | 2017-03-20 21:28:49 -0400 | [diff] [blame] | 277 | p = prefix_filename(prefix, p); |
Junio C Hamano | 3b069b1 | 2012-06-27 12:05:52 -0700 | [diff] [blame] | 278 | paths[i] = p; |
Junio C Hamano | 0569e9b | 2008-05-23 22:28:56 -0700 | [diff] [blame] | 279 | } |
Junio C Hamano | c9e1f2c | 2015-03-25 16:11:39 -0700 | [diff] [blame] | 280 | |
| 281 | fixup_paths(paths, &replacement); |
| 282 | |
Michael Spang | d61027b | 2009-02-18 01:48:06 -0500 | [diff] [blame] | 283 | revs->diffopt.skip_stat_unmatch = 1; |
Johannes Sixt | 5d83f9c | 2009-03-25 18:19:46 +0100 | [diff] [blame] | 284 | if (!revs->diffopt.output_format) |
| 285 | revs->diffopt.output_format = DIFF_FORMAT_PATCH; |
Junio C Hamano | 0569e9b | 2008-05-23 22:28:56 -0700 | [diff] [blame] | 286 | |
Brandon Williams | 0d1e0e7 | 2017-10-31 11:19:11 -0700 | [diff] [blame] | 287 | revs->diffopt.flags.no_index = 1; |
Junio C Hamano | 0569e9b | 2008-05-23 22:28:56 -0700 | [diff] [blame] | 288 | |
Brandon Williams | 0d1e0e7 | 2017-10-31 11:19:11 -0700 | [diff] [blame] | 289 | revs->diffopt.flags.relative_name = 1; |
Jeff King | 7d8930d | 2016-09-12 20:23:32 -0700 | [diff] [blame] | 290 | revs->diffopt.prefix = prefix; |
| 291 | |
Junio C Hamano | 0569e9b | 2008-05-23 22:28:56 -0700 | [diff] [blame] | 292 | revs->max_count = -2; |
Thomas Rast | 2845265 | 2012-08-03 14:16:24 +0200 | [diff] [blame] | 293 | diff_setup_done(&revs->diffopt); |
Junio C Hamano | 0569e9b | 2008-05-23 22:28:56 -0700 | [diff] [blame] | 294 | |
Jeff King | af63b54 | 2012-06-15 16:32:55 -0400 | [diff] [blame] | 295 | setup_diff_pager(&revs->diffopt); |
Brandon Williams | 0d1e0e7 | 2017-10-31 11:19:11 -0700 | [diff] [blame] | 296 | revs->diffopt.flags.exit_with_status = 1; |
Jeff King | af63b54 | 2012-06-15 16:32:55 -0400 | [diff] [blame] | 297 | |
Junio C Hamano | c20f592 | 2012-06-27 11:51:15 -0700 | [diff] [blame] | 298 | if (queue_diff(&revs->diffopt, paths[0], paths[1])) |
Nguyễn Thái Ngọc Duy | 16bb3d7 | 2019-03-24 15:20:13 +0700 | [diff] [blame] | 299 | return 1; |
Junio C Hamano | a5a818e | 2008-08-18 20:08:09 -0700 | [diff] [blame] | 300 | diff_set_mnemonic_prefix(&revs->diffopt, "1/", "2/"); |
Junio C Hamano | 0569e9b | 2008-05-23 22:28:56 -0700 | [diff] [blame] | 301 | diffcore_std(&revs->diffopt); |
| 302 | diff_flush(&revs->diffopt); |
| 303 | |
Junio C Hamano | c9e1f2c | 2015-03-25 16:11:39 -0700 | [diff] [blame] | 304 | strbuf_release(&replacement); |
| 305 | |
Junio C Hamano | 0569e9b | 2008-05-23 22:28:56 -0700 | [diff] [blame] | 306 | /* |
| 307 | * The return code for --no-index imitates diff(1): |
| 308 | * 0 = no changes, 1 = changes, else error |
| 309 | */ |
Nguyễn Thái Ngọc Duy | 16bb3d7 | 2019-03-24 15:20:13 +0700 | [diff] [blame] | 310 | return diff_result_code(&revs->diffopt, 0); |
Junio C Hamano | 0569e9b | 2008-05-23 22:28:56 -0700 | [diff] [blame] | 311 | } |