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" |
Johannes Schindelin | c455c87 | 2008-07-21 19:03:49 +0100 | [diff] [blame] | 17 | #include "string-list.h" |
Brian Bourn | fd3aeea | 2014-03-19 11:58:22 -0400 | [diff] [blame] | 18 | #include "dir.h" |
Junio C Hamano | 0569e9b | 2008-05-23 22:28:56 -0700 | [diff] [blame] | 19 | |
Brian Bourn | 9daf0ef | 2014-03-19 11:58:21 -0400 | [diff] [blame] | 20 | 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] | 21 | { |
| 22 | DIR *dir; |
| 23 | struct dirent *e; |
| 24 | |
| 25 | if (!(dir = opendir(path))) |
| 26 | return error("Could not open directory %s", path); |
| 27 | |
| 28 | while ((e = readdir(dir))) |
Brian Bourn | fd3aeea | 2014-03-19 11:58:22 -0400 | [diff] [blame] | 29 | if (!is_dot_or_dotdot(e->d_name)) |
Julian Phillips | 78a395d | 2010-06-26 00:41:35 +0100 | [diff] [blame] | 30 | string_list_insert(list, e->d_name); |
Junio C Hamano | 0569e9b | 2008-05-23 22:28:56 -0700 | [diff] [blame] | 31 | |
| 32 | closedir(dir); |
| 33 | return 0; |
| 34 | } |
| 35 | |
Junio C Hamano | 4682d85 | 2012-06-27 20:14:47 -0700 | [diff] [blame] | 36 | /* |
| 37 | * This should be "(standard input)" or something, but it will |
| 38 | * probably expose many more breakages in the way no-index code |
| 39 | * is bolted onto the diff callchain. |
| 40 | */ |
| 41 | static const char file_from_standard_input[] = "-"; |
| 42 | |
Junio C Hamano | 0569e9b | 2008-05-23 22:28:56 -0700 | [diff] [blame] | 43 | static int get_mode(const char *path, int *mode) |
| 44 | { |
| 45 | struct stat st; |
| 46 | |
| 47 | if (!path || !strcmp(path, "/dev/null")) |
| 48 | *mode = 0; |
Jonathan Nieder | 380395d | 2013-05-02 20:26:08 +0100 | [diff] [blame] | 49 | #ifdef GIT_WINDOWS_NATIVE |
Johannes Schindelin | 36adb4a | 2009-03-07 16:51:33 +0100 | [diff] [blame] | 50 | else if (!strcasecmp(path, "nul")) |
| 51 | *mode = 0; |
| 52 | #endif |
Junio C Hamano | 4682d85 | 2012-06-27 20:14:47 -0700 | [diff] [blame] | 53 | else if (path == file_from_standard_input) |
Junio C Hamano | 0569e9b | 2008-05-23 22:28:56 -0700 | [diff] [blame] | 54 | *mode = create_ce_mode(0666); |
Johannes Schindelin | 418566b | 2009-01-29 17:30:51 +0100 | [diff] [blame] | 55 | else if (lstat(path, &st)) |
Junio C Hamano | 0569e9b | 2008-05-23 22:28:56 -0700 | [diff] [blame] | 56 | return error("Could not access '%s'", path); |
| 57 | else |
| 58 | *mode = st.st_mode; |
| 59 | return 0; |
| 60 | } |
| 61 | |
Junio C Hamano | 4682d85 | 2012-06-27 20:14:47 -0700 | [diff] [blame] | 62 | static int populate_from_stdin(struct diff_filespec *s) |
| 63 | { |
| 64 | struct strbuf buf = STRBUF_INIT; |
| 65 | size_t size = 0; |
| 66 | |
| 67 | if (strbuf_read(&buf, 0, 0) < 0) |
| 68 | return error("error while reading from stdin %s", |
| 69 | strerror(errno)); |
| 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); |
Junio C Hamano | 3b75314 | 2012-08-27 11:54:28 -0700 | [diff] [blame] | 86 | fill_filespec(s, null_sha1, 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); |
| 139 | if (buffer1.len && buffer1.buf[buffer1.len - 1] != '/') |
| 140 | strbuf_addch(&buffer1, '/'); |
Bobby Powers | f3999e0 | 2012-05-16 10:28:31 -0400 | [diff] [blame] | 141 | len1 = buffer1.len; |
Junio C Hamano | 0569e9b | 2008-05-23 22:28:56 -0700 | [diff] [blame] | 142 | } |
| 143 | |
| 144 | if (name2) { |
Junio C Hamano | 875b91b | 2012-04-25 12:37:38 -0700 | [diff] [blame] | 145 | strbuf_addstr(&buffer2, name2); |
| 146 | if (buffer2.len && buffer2.buf[buffer2.len - 1] != '/') |
| 147 | strbuf_addch(&buffer2, '/'); |
Bobby Powers | f3999e0 | 2012-05-16 10:28:31 -0400 | [diff] [blame] | 148 | len2 = buffer2.len; |
Junio C Hamano | 0569e9b | 2008-05-23 22:28:56 -0700 | [diff] [blame] | 149 | } |
| 150 | |
| 151 | for (i1 = i2 = 0; !ret && (i1 < p1.nr || i2 < p2.nr); ) { |
| 152 | const char *n1, *n2; |
| 153 | int comp; |
| 154 | |
Bobby Powers | f3999e0 | 2012-05-16 10:28:31 -0400 | [diff] [blame] | 155 | strbuf_setlen(&buffer1, len1); |
| 156 | strbuf_setlen(&buffer2, len2); |
| 157 | |
Junio C Hamano | 0569e9b | 2008-05-23 22:28:56 -0700 | [diff] [blame] | 158 | if (i1 == p1.nr) |
| 159 | comp = 1; |
| 160 | else if (i2 == p2.nr) |
| 161 | comp = -1; |
| 162 | else |
Junio C Hamano | 875b91b | 2012-04-25 12:37:38 -0700 | [diff] [blame] | 163 | comp = strcmp(p1.items[i1].string, p2.items[i2].string); |
Junio C Hamano | 0569e9b | 2008-05-23 22:28:56 -0700 | [diff] [blame] | 164 | |
| 165 | if (comp > 0) |
| 166 | n1 = NULL; |
| 167 | else { |
Junio C Hamano | 875b91b | 2012-04-25 12:37:38 -0700 | [diff] [blame] | 168 | strbuf_addstr(&buffer1, p1.items[i1++].string); |
| 169 | n1 = buffer1.buf; |
Junio C Hamano | 0569e9b | 2008-05-23 22:28:56 -0700 | [diff] [blame] | 170 | } |
| 171 | |
| 172 | if (comp < 0) |
| 173 | n2 = NULL; |
| 174 | else { |
Junio C Hamano | 875b91b | 2012-04-25 12:37:38 -0700 | [diff] [blame] | 175 | strbuf_addstr(&buffer2, p2.items[i2++].string); |
| 176 | n2 = buffer2.buf; |
Junio C Hamano | 0569e9b | 2008-05-23 22:28:56 -0700 | [diff] [blame] | 177 | } |
| 178 | |
| 179 | ret = queue_diff(o, n1, n2); |
| 180 | } |
Johannes Schindelin | c455c87 | 2008-07-21 19:03:49 +0100 | [diff] [blame] | 181 | string_list_clear(&p1, 0); |
| 182 | string_list_clear(&p2, 0); |
Bobby Powers | 176a335 | 2012-05-16 10:50:31 -0400 | [diff] [blame] | 183 | strbuf_release(&buffer1); |
| 184 | strbuf_release(&buffer2); |
Junio C Hamano | 0569e9b | 2008-05-23 22:28:56 -0700 | [diff] [blame] | 185 | |
| 186 | return ret; |
| 187 | } else { |
| 188 | struct diff_filespec *d1, *d2; |
| 189 | |
| 190 | if (DIFF_OPT_TST(o, REVERSE_DIFF)) { |
| 191 | unsigned tmp; |
| 192 | const char *tmp_c; |
| 193 | tmp = mode1; mode1 = mode2; mode2 = tmp; |
| 194 | tmp_c = name1; name1 = name2; name2 = tmp_c; |
| 195 | } |
| 196 | |
Junio C Hamano | 4682d85 | 2012-06-27 20:14:47 -0700 | [diff] [blame] | 197 | d1 = noindex_filespec(name1, mode1); |
| 198 | d2 = noindex_filespec(name2, mode2); |
Junio C Hamano | 0569e9b | 2008-05-23 22:28:56 -0700 | [diff] [blame] | 199 | diff_queue(&diff_queued_diff, d1, d2); |
| 200 | return 0; |
| 201 | } |
| 202 | } |
| 203 | |
Junio C Hamano | c9e1f2c | 2015-03-25 16:11:39 -0700 | [diff] [blame] | 204 | /* append basename of F to D */ |
| 205 | static void append_basename(struct strbuf *path, const char *dir, const char *file) |
| 206 | { |
| 207 | const char *tail = strrchr(file, '/'); |
| 208 | |
| 209 | strbuf_addstr(path, dir); |
| 210 | while (path->len && path->buf[path->len - 1] == '/') |
| 211 | path->len--; |
| 212 | strbuf_addch(path, '/'); |
| 213 | strbuf_addstr(path, tail ? tail + 1 : file); |
| 214 | } |
| 215 | |
| 216 | /* |
| 217 | * DWIM "diff D F" into "diff D/F F" and "diff F D" into "diff F D/F" |
| 218 | * Note that we append the basename of F to D/, so "diff a/b/file D" |
| 219 | * becomes "diff a/b/file D/file", not "diff a/b/file D/a/b/file". |
| 220 | */ |
| 221 | static void fixup_paths(const char **path, struct strbuf *replacement) |
| 222 | { |
| 223 | unsigned int isdir0, isdir1; |
| 224 | |
| 225 | if (path[0] == file_from_standard_input || |
| 226 | path[1] == file_from_standard_input) |
| 227 | return; |
| 228 | isdir0 = is_directory(path[0]); |
| 229 | isdir1 = is_directory(path[1]); |
| 230 | if (isdir0 == isdir1) |
| 231 | return; |
| 232 | if (isdir0) { |
| 233 | append_basename(replacement, path[0], path[1]); |
| 234 | path[0] = replacement->buf; |
| 235 | } else { |
| 236 | append_basename(replacement, path[1], path[0]); |
| 237 | path[1] = replacement->buf; |
| 238 | } |
| 239 | } |
| 240 | |
Junio C Hamano | 0569e9b | 2008-05-23 22:28:56 -0700 | [diff] [blame] | 241 | void diff_no_index(struct rev_info *revs, |
| 242 | int argc, const char **argv, |
Thomas Gummerer | 470faf9 | 2013-12-11 10:58:42 +0100 | [diff] [blame] | 243 | const char *prefix) |
Junio C Hamano | 0569e9b | 2008-05-23 22:28:56 -0700 | [diff] [blame] | 244 | { |
Junio C Hamano | 3b069b1 | 2012-06-27 12:05:52 -0700 | [diff] [blame] | 245 | int i, prefixlen; |
Junio C Hamano | c20f592 | 2012-06-27 11:51:15 -0700 | [diff] [blame] | 246 | const char *paths[2]; |
Junio C Hamano | c9e1f2c | 2015-03-25 16:11:39 -0700 | [diff] [blame] | 247 | struct strbuf replacement = STRBUF_INIT; |
Junio C Hamano | 0569e9b | 2008-05-23 22:28:56 -0700 | [diff] [blame] | 248 | |
Junio C Hamano | 0569e9b | 2008-05-23 22:28:56 -0700 | [diff] [blame] | 249 | diff_setup(&revs->diffopt); |
Junio C Hamano | 0569e9b | 2008-05-23 22:28:56 -0700 | [diff] [blame] | 250 | for (i = 1; i < argc - 2; ) { |
| 251 | int j; |
| 252 | if (!strcmp(argv[i], "--no-index")) |
| 253 | i++; |
Thomas Rast | e423ffd | 2009-01-06 19:53:32 +0100 | [diff] [blame] | 254 | else if (!strcmp(argv[i], "--")) |
| 255 | i++; |
Junio C Hamano | 0569e9b | 2008-05-23 22:28:56 -0700 | [diff] [blame] | 256 | else { |
| 257 | j = diff_opt_parse(&revs->diffopt, argv + i, argc - i); |
Junio C Hamano | ad1c3fb | 2014-03-31 11:47:17 -0700 | [diff] [blame] | 258 | if (j <= 0) |
Junio C Hamano | 0569e9b | 2008-05-23 22:28:56 -0700 | [diff] [blame] | 259 | die("invalid diff option/value: %s", argv[i]); |
| 260 | i += j; |
| 261 | } |
| 262 | } |
| 263 | |
Junio C Hamano | 3b069b1 | 2012-06-27 12:05:52 -0700 | [diff] [blame] | 264 | prefixlen = prefix ? strlen(prefix) : 0; |
| 265 | for (i = 0; i < 2; i++) { |
| 266 | const char *p = argv[argc - 2 + i]; |
| 267 | if (!strcmp(p, "-")) |
Junio C Hamano | 0569e9b | 2008-05-23 22:28:56 -0700 | [diff] [blame] | 268 | /* |
Junio C Hamano | 3b069b1 | 2012-06-27 12:05:52 -0700 | [diff] [blame] | 269 | * stdin should be spelled as "-"; if you have |
| 270 | * path that is "-", spell it as "./-". |
Junio C Hamano | 0569e9b | 2008-05-23 22:28:56 -0700 | [diff] [blame] | 271 | */ |
Junio C Hamano | 4682d85 | 2012-06-27 20:14:47 -0700 | [diff] [blame] | 272 | p = file_from_standard_input; |
Junio C Hamano | 3b069b1 | 2012-06-27 12:05:52 -0700 | [diff] [blame] | 273 | else if (prefixlen) |
| 274 | p = xstrdup(prefix_filename(prefix, prefixlen, p)); |
| 275 | paths[i] = p; |
Junio C Hamano | 0569e9b | 2008-05-23 22:28:56 -0700 | [diff] [blame] | 276 | } |
Junio C Hamano | c9e1f2c | 2015-03-25 16:11:39 -0700 | [diff] [blame] | 277 | |
| 278 | fixup_paths(paths, &replacement); |
| 279 | |
Michael Spang | d61027b | 2009-02-18 01:48:06 -0500 | [diff] [blame] | 280 | revs->diffopt.skip_stat_unmatch = 1; |
Johannes Sixt | 5d83f9c | 2009-03-25 18:19:46 +0100 | [diff] [blame] | 281 | if (!revs->diffopt.output_format) |
| 282 | revs->diffopt.output_format = DIFF_FORMAT_PATCH; |
Junio C Hamano | 0569e9b | 2008-05-23 22:28:56 -0700 | [diff] [blame] | 283 | |
Junio C Hamano | 0569e9b | 2008-05-23 22:28:56 -0700 | [diff] [blame] | 284 | DIFF_OPT_SET(&revs->diffopt, NO_INDEX); |
| 285 | |
| 286 | revs->max_count = -2; |
Thomas Rast | 2845265 | 2012-08-03 14:16:24 +0200 | [diff] [blame] | 287 | diff_setup_done(&revs->diffopt); |
Junio C Hamano | 0569e9b | 2008-05-23 22:28:56 -0700 | [diff] [blame] | 288 | |
Jeff King | af63b54 | 2012-06-15 16:32:55 -0400 | [diff] [blame] | 289 | setup_diff_pager(&revs->diffopt); |
| 290 | DIFF_OPT_SET(&revs->diffopt, EXIT_WITH_STATUS); |
| 291 | |
Junio C Hamano | c20f592 | 2012-06-27 11:51:15 -0700 | [diff] [blame] | 292 | if (queue_diff(&revs->diffopt, paths[0], paths[1])) |
Junio C Hamano | 0569e9b | 2008-05-23 22:28:56 -0700 | [diff] [blame] | 293 | exit(1); |
Junio C Hamano | a5a818e | 2008-08-18 20:08:09 -0700 | [diff] [blame] | 294 | diff_set_mnemonic_prefix(&revs->diffopt, "1/", "2/"); |
Junio C Hamano | 0569e9b | 2008-05-23 22:28:56 -0700 | [diff] [blame] | 295 | diffcore_std(&revs->diffopt); |
| 296 | diff_flush(&revs->diffopt); |
| 297 | |
Junio C Hamano | c9e1f2c | 2015-03-25 16:11:39 -0700 | [diff] [blame] | 298 | strbuf_release(&replacement); |
| 299 | |
Junio C Hamano | 0569e9b | 2008-05-23 22:28:56 -0700 | [diff] [blame] | 300 | /* |
| 301 | * The return code for --no-index imitates diff(1): |
| 302 | * 0 = no changes, 1 = changes, else error |
| 303 | */ |
Tim Henigan | 304970d | 2012-06-21 14:09:51 -0400 | [diff] [blame] | 304 | exit(diff_result_code(&revs->diffopt, 0)); |
Junio C Hamano | 0569e9b | 2008-05-23 22:28:56 -0700 | [diff] [blame] | 305 | } |