David Reiss | ae299be | 2008-05-19 23:48:54 -0700 | [diff] [blame] | 1 | #include "cache.h" |
Michael Haggerty | 31171d9 | 2012-10-28 17:16:24 +0100 | [diff] [blame] | 2 | #include "string-list.h" |
David Reiss | ae299be | 2008-05-19 23:48:54 -0700 | [diff] [blame] | 3 | |
Michael Haggerty | 9e2326c | 2012-10-28 17:16:25 +0100 | [diff] [blame] | 4 | /* |
| 5 | * A "string_list_each_func_t" function that normalizes an entry from |
| 6 | * GIT_CEILING_DIRECTORIES. If the path is unusable for some reason, |
| 7 | * die with an explanation. |
| 8 | */ |
| 9 | static int normalize_ceiling_entry(struct string_list_item *item, void *unused) |
| 10 | { |
| 11 | const char *ceil = item->string; |
| 12 | int len = strlen(ceil); |
| 13 | char buf[PATH_MAX+1]; |
| 14 | |
| 15 | if (len == 0) |
| 16 | die("Empty path is not supported"); |
| 17 | if (len > PATH_MAX) |
| 18 | die("Path \"%s\" is too long", ceil); |
| 19 | if (!is_absolute_path(ceil)) |
| 20 | die("Path \"%s\" is not absolute", ceil); |
| 21 | if (normalize_path_copy(buf, ceil) < 0) |
| 22 | die("Path \"%s\" could not be normalized", ceil); |
| 23 | len = strlen(buf); |
Michael Haggerty | 9e2326c | 2012-10-28 17:16:25 +0100 | [diff] [blame] | 24 | free(item->string); |
| 25 | item->string = xstrdup(buf); |
| 26 | return 1; |
| 27 | } |
| 28 | |
Jiang Xin | 203439b | 2013-06-25 23:53:42 +0800 | [diff] [blame] | 29 | static void normalize_argv_string(const char **var, const char *input) |
| 30 | { |
| 31 | if (!strcmp(input, "<null>")) |
| 32 | *var = NULL; |
| 33 | else if (!strcmp(input, "<empty>")) |
| 34 | *var = ""; |
| 35 | else |
| 36 | *var = input; |
| 37 | |
| 38 | if (*var && (**var == '<' || **var == '(')) |
| 39 | die("Bad value: %s\n", input); |
| 40 | } |
| 41 | |
David Reiss | ae299be | 2008-05-19 23:48:54 -0700 | [diff] [blame] | 42 | int main(int argc, char **argv) |
| 43 | { |
Johannes Sixt | f42302b | 2009-02-07 16:08:30 +0100 | [diff] [blame] | 44 | if (argc == 3 && !strcmp(argv[1], "normalize_path_copy")) { |
Johannes Schindelin | b8469ad | 2009-01-28 00:07:36 +0100 | [diff] [blame] | 45 | char *buf = xmalloc(PATH_MAX + 1); |
Johannes Sixt | f42302b | 2009-02-07 16:08:30 +0100 | [diff] [blame] | 46 | int rv = normalize_path_copy(buf, argv[2]); |
| 47 | if (rv) |
| 48 | buf = "++failed++"; |
David Reiss | ae299be | 2008-05-19 23:48:54 -0700 | [diff] [blame] | 49 | puts(buf); |
Johannes Sixt | 2cd85c4 | 2009-02-07 16:08:27 +0100 | [diff] [blame] | 50 | return 0; |
David Reiss | ae299be | 2008-05-19 23:48:54 -0700 | [diff] [blame] | 51 | } |
| 52 | |
Carlos Martín Nieto | e2a57aa | 2011-03-17 12:26:46 +0100 | [diff] [blame] | 53 | if (argc >= 2 && !strcmp(argv[1], "real_path")) { |
David Reiss | d553e73 | 2008-05-19 23:49:00 -0700 | [diff] [blame] | 54 | while (argc > 2) { |
Carlos Martín Nieto | e2a57aa | 2011-03-17 12:26:46 +0100 | [diff] [blame] | 55 | puts(real_path(argv[2])); |
David Reiss | d553e73 | 2008-05-19 23:49:00 -0700 | [diff] [blame] | 56 | argc--; |
| 57 | argv++; |
| 58 | } |
Johannes Sixt | 2cd85c4 | 2009-02-07 16:08:27 +0100 | [diff] [blame] | 59 | return 0; |
David Reiss | d553e73 | 2008-05-19 23:49:00 -0700 | [diff] [blame] | 60 | } |
| 61 | |
Michael Haggerty | 87a246e | 2011-08-04 06:47:47 +0200 | [diff] [blame] | 62 | if (argc >= 2 && !strcmp(argv[1], "absolute_path")) { |
| 63 | while (argc > 2) { |
| 64 | puts(absolute_path(argv[2])); |
| 65 | argc--; |
| 66 | argv++; |
| 67 | } |
| 68 | return 0; |
| 69 | } |
| 70 | |
David Reiss | 0454dd9 | 2008-05-19 23:49:26 -0700 | [diff] [blame] | 71 | if (argc == 4 && !strcmp(argv[1], "longest_ancestor_length")) { |
Michael Haggerty | 31171d9 | 2012-10-28 17:16:24 +0100 | [diff] [blame] | 72 | int len; |
| 73 | struct string_list ceiling_dirs = STRING_LIST_INIT_DUP; |
Michael Haggerty | 9e2326c | 2012-10-28 17:16:25 +0100 | [diff] [blame] | 74 | char *path = xstrdup(argv[2]); |
Michael Haggerty | 31171d9 | 2012-10-28 17:16:24 +0100 | [diff] [blame] | 75 | |
Michael Haggerty | 9e2326c | 2012-10-28 17:16:25 +0100 | [diff] [blame] | 76 | /* |
| 77 | * We have to normalize the arguments because under |
| 78 | * Windows, bash mangles arguments that look like |
| 79 | * absolute POSIX paths or colon-separate lists of |
| 80 | * absolute POSIX paths into DOS paths (e.g., |
| 81 | * "/foo:/foo/bar" might be converted to |
| 82 | * "D:\Src\msysgit\foo;D:\Src\msysgit\foo\bar"), |
| 83 | * whereas longest_ancestor_length() requires paths |
| 84 | * that use forward slashes. |
| 85 | */ |
| 86 | if (normalize_path_copy(path, path)) |
| 87 | die("Path \"%s\" could not be normalized", argv[2]); |
Michael Haggerty | 31171d9 | 2012-10-28 17:16:24 +0100 | [diff] [blame] | 88 | string_list_split(&ceiling_dirs, argv[3], PATH_SEP, -1); |
Michael Haggerty | 9e2326c | 2012-10-28 17:16:25 +0100 | [diff] [blame] | 89 | filter_string_list(&ceiling_dirs, 0, |
| 90 | normalize_ceiling_entry, NULL); |
| 91 | len = longest_ancestor_length(path, &ceiling_dirs); |
Michael Haggerty | 31171d9 | 2012-10-28 17:16:24 +0100 | [diff] [blame] | 92 | string_list_clear(&ceiling_dirs, 0); |
Michael Haggerty | 9e2326c | 2012-10-28 17:16:25 +0100 | [diff] [blame] | 93 | free(path); |
David Reiss | 0454dd9 | 2008-05-19 23:49:26 -0700 | [diff] [blame] | 94 | printf("%d\n", len); |
Johannes Sixt | 2cd85c4 | 2009-02-07 16:08:27 +0100 | [diff] [blame] | 95 | return 0; |
David Reiss | 0454dd9 | 2008-05-19 23:49:26 -0700 | [diff] [blame] | 96 | } |
| 97 | |
Michael Haggerty | 9e81372 | 2011-08-04 06:47:48 +0200 | [diff] [blame] | 98 | if (argc >= 4 && !strcmp(argv[1], "prefix_path")) { |
| 99 | char *prefix = argv[2]; |
| 100 | int prefix_len = strlen(prefix); |
| 101 | int nongit_ok; |
| 102 | setup_git_directory_gently(&nongit_ok); |
| 103 | while (argc > 3) { |
| 104 | puts(prefix_path(prefix, prefix_len, argv[3])); |
| 105 | argc--; |
| 106 | argv++; |
| 107 | } |
| 108 | return 0; |
| 109 | } |
| 110 | |
Johannes Schindelin | 4fcc86b | 2009-02-19 20:10:49 +0100 | [diff] [blame] | 111 | if (argc == 4 && !strcmp(argv[1], "strip_path_suffix")) { |
| 112 | char *prefix = strip_path_suffix(argv[2], argv[3]); |
| 113 | printf("%s\n", prefix ? prefix : "(null)"); |
| 114 | return 0; |
| 115 | } |
| 116 | |
Sebastian Schuberth | 7ffd18f | 2013-10-10 22:49:43 +0200 | [diff] [blame] | 117 | if (argc == 3 && !strcmp(argv[1], "print_path")) { |
Jiang Xin | abd4284 | 2013-06-25 23:53:57 +0800 | [diff] [blame] | 118 | puts(argv[2]); |
| 119 | return 0; |
| 120 | } |
| 121 | |
Jiang Xin | 203439b | 2013-06-25 23:53:42 +0800 | [diff] [blame] | 122 | if (argc == 4 && !strcmp(argv[1], "relative_path")) { |
Jiang Xin | e02ca72 | 2013-06-25 23:53:43 +0800 | [diff] [blame] | 123 | struct strbuf sb = STRBUF_INIT; |
Jiang Xin | 203439b | 2013-06-25 23:53:42 +0800 | [diff] [blame] | 124 | const char *in, *prefix, *rel; |
| 125 | normalize_argv_string(&in, argv[2]); |
| 126 | normalize_argv_string(&prefix, argv[3]); |
Jiang Xin | e02ca72 | 2013-06-25 23:53:43 +0800 | [diff] [blame] | 127 | rel = relative_path(in, prefix, &sb); |
Jiang Xin | 203439b | 2013-06-25 23:53:42 +0800 | [diff] [blame] | 128 | if (!rel) |
| 129 | puts("(null)"); |
| 130 | else |
| 131 | puts(strlen(rel) > 0 ? rel : "(empty)"); |
Jiang Xin | e02ca72 | 2013-06-25 23:53:43 +0800 | [diff] [blame] | 132 | strbuf_release(&sb); |
Jiang Xin | 203439b | 2013-06-25 23:53:42 +0800 | [diff] [blame] | 133 | return 0; |
| 134 | } |
| 135 | |
Johannes Sixt | 2cd85c4 | 2009-02-07 16:08:27 +0100 | [diff] [blame] | 136 | fprintf(stderr, "%s: unknown function name: %s\n", argv[0], |
| 137 | argv[1] ? argv[1] : "(there was none)"); |
| 138 | return 1; |
David Reiss | ae299be | 2008-05-19 23:48:54 -0700 | [diff] [blame] | 139 | } |