5 #include "refs/refs-internal.h"
6 #include "object-name.h"
7 #include "object-store-ll.h"
9 #include "string-list.h"
10 #include "parse-options.h"
12 static const char * const show_ref_usage
[] = {
13 N_("git show-ref [--head] [-d | --dereference]\n"
14 " [-s | --hash[=<n>]] [--abbrev[=<n>]] [--tags]\n"
15 " [--heads] [--] [<pattern>...]"),
16 N_("git show-ref --verify [-q | --quiet] [-d | --dereference]\n"
17 " [-s | --hash[=<n>]] [--abbrev[=<n>]]\n"
19 N_("git show-ref --exclude-existing[=<pattern>]"),
20 N_("git show-ref --exists <ref>"),
24 struct show_one_options
{
31 static void show_one(const struct show_one_options
*opts
,
32 const char *refname
, const struct object_id
*oid
)
35 struct object_id peeled
;
37 if (!repo_has_object_file(the_repository
, oid
))
38 die("git show-ref: bad ref %s (%s)", refname
,
44 hex
= repo_find_unique_abbrev(the_repository
, oid
, opts
->abbrev
);
48 printf("%s %s\n", hex
, refname
);
50 if (!opts
->deref_tags
)
53 if (!peel_iterated_oid(the_repository
, oid
, &peeled
)) {
54 hex
= repo_find_unique_abbrev(the_repository
, &peeled
, opts
->abbrev
);
55 printf("%s %s^{}\n", hex
, refname
);
59 struct show_ref_data
{
60 const struct show_one_options
*show_one_opts
;
61 const char **patterns
;
66 static int show_ref(const char *refname
, const struct object_id
*oid
,
67 int flag UNUSED
, void *cbdata
)
69 struct show_ref_data
*data
= cbdata
;
71 if (data
->show_head
&& !strcmp(refname
, "HEAD"))
75 int reflen
= strlen(refname
);
76 const char **p
= data
->patterns
, *m
;
77 while ((m
= *p
++) != NULL
) {
81 if (memcmp(m
, refname
+ reflen
- len
, len
))
85 if (refname
[reflen
- len
- 1] == '/')
94 show_one(data
->show_one_opts
, refname
, oid
);
99 static int add_existing(const char *refname
,
100 const struct object_id
*oid UNUSED
,
101 int flag UNUSED
, void *cbdata
)
103 struct string_list
*list
= (struct string_list
*)cbdata
;
104 string_list_insert(list
, refname
);
108 struct exclude_existing_options
{
110 * We need an explicit `enabled` field because it is perfectly valid
111 * for `pattern` to be `NULL` even if `--exclude-existing` was given.
118 * read "^(?:<anything>\s)?<refname>(?:\^\{\})?$" from the standard input,
120 * (1) strip "^{}" at the end of line if any;
121 * (2) ignore if match is provided and does not head-match refname;
122 * (3) warn if refname is not a well-formed refname and skip;
123 * (4) ignore if refname is a ref that exists in the local repository;
124 * (5) otherwise output the line.
126 static int cmd_show_ref__exclude_existing(const struct exclude_existing_options
*opts
)
128 struct string_list existing_refs
= STRING_LIST_INIT_DUP
;
130 int patternlen
= opts
->pattern
? strlen(opts
->pattern
) : 0;
132 refs_for_each_ref(get_main_ref_store(the_repository
), add_existing
,
134 while (fgets(buf
, sizeof(buf
), stdin
)) {
136 int len
= strlen(buf
);
138 if (len
> 0 && buf
[len
- 1] == '\n')
140 if (3 <= len
&& !strcmp(buf
+ len
- 3, "^{}")) {
144 for (ref
= buf
+ len
; buf
< ref
; ref
--)
145 if (isspace(ref
[-1]))
148 int reflen
= buf
+ len
- ref
;
149 if (reflen
< patternlen
)
151 if (strncmp(ref
, opts
->pattern
, patternlen
))
154 if (check_refname_format(ref
, 0)) {
155 warning("ref '%s' ignored", ref
);
158 if (!string_list_has_string(&existing_refs
, ref
)) {
163 string_list_clear(&existing_refs
, 0);
167 static int cmd_show_ref__verify(const struct show_one_options
*show_one_opts
,
171 die("--verify requires a reference");
174 struct object_id oid
;
176 if ((starts_with(*refs
, "refs/") || refname_is_safe(*refs
)) &&
177 !refs_read_ref(get_main_ref_store(the_repository
), *refs
, &oid
)) {
178 show_one(show_one_opts
, *refs
, &oid
);
180 else if (!show_one_opts
->quiet
)
181 die("'%s' - not a valid ref", *refs
);
190 struct patterns_options
{
196 static int cmd_show_ref__patterns(const struct patterns_options
*opts
,
197 const struct show_one_options
*show_one_opts
,
198 const char **patterns
)
200 struct show_ref_data show_ref_data
= {
201 .show_one_opts
= show_one_opts
,
202 .show_head
= opts
->show_head
,
205 if (patterns
&& *patterns
)
206 show_ref_data
.patterns
= patterns
;
209 refs_head_ref(get_main_ref_store(the_repository
), show_ref
,
211 if (opts
->heads_only
|| opts
->tags_only
) {
212 if (opts
->heads_only
)
213 refs_for_each_fullref_in(get_main_ref_store(the_repository
),
215 show_ref
, &show_ref_data
);
217 refs_for_each_fullref_in(get_main_ref_store(the_repository
),
218 "refs/tags/", NULL
, show_ref
,
221 refs_for_each_ref(get_main_ref_store(the_repository
),
222 show_ref
, &show_ref_data
);
224 if (!show_ref_data
.found_match
)
230 static int cmd_show_ref__exists(const char **refs
)
232 struct strbuf unused_referent
= STRBUF_INIT
;
233 struct object_id unused_oid
;
234 unsigned int unused_type
;
235 int failure_errno
= 0;
240 die("--exists requires a reference");
243 die("--exists requires exactly one reference");
245 if (refs_read_raw_ref(get_main_ref_store(the_repository
), ref
,
246 &unused_oid
, &unused_referent
, &unused_type
,
248 if (failure_errno
== ENOENT
|| failure_errno
== EISDIR
) {
249 error(_("reference does not exist"));
252 errno
= failure_errno
;
253 error_errno(_("failed to look up reference"));
261 strbuf_release(&unused_referent
);
265 static int hash_callback(const struct option
*opt
, const char *arg
, int unset
)
267 struct show_one_options
*opts
= opt
->value
;
268 struct option abbrev_opt
= *opt
;
271 /* Use full length SHA1 if no argument */
275 abbrev_opt
.value
= &opts
->abbrev
;
276 return parse_opt_abbrev_cb(&abbrev_opt
, arg
, unset
);
279 static int exclude_existing_callback(const struct option
*opt
, const char *arg
,
282 struct exclude_existing_options
*opts
= opt
->value
;
283 BUG_ON_OPT_NEG(unset
);
289 int cmd_show_ref(int argc
, const char **argv
, const char *prefix
)
291 struct exclude_existing_options exclude_existing_opts
= {0};
292 struct patterns_options patterns_opts
= {0};
293 struct show_one_options show_one_opts
= {0};
294 int verify
= 0, exists
= 0;
295 const struct option show_ref_options
[] = {
296 OPT_BOOL(0, "tags", &patterns_opts
.tags_only
, N_("only show tags (can be combined with heads)")),
297 OPT_BOOL(0, "heads", &patterns_opts
.heads_only
, N_("only show heads (can be combined with tags)")),
298 OPT_BOOL(0, "exists", &exists
, N_("check for reference existence without resolving")),
299 OPT_BOOL(0, "verify", &verify
, N_("stricter reference checking, "
300 "requires exact ref path")),
301 OPT_HIDDEN_BOOL('h', NULL
, &patterns_opts
.show_head
,
302 N_("show the HEAD reference, even if it would be filtered out")),
303 OPT_BOOL(0, "head", &patterns_opts
.show_head
,
304 N_("show the HEAD reference, even if it would be filtered out")),
305 OPT_BOOL('d', "dereference", &show_one_opts
.deref_tags
,
306 N_("dereference tags into object IDs")),
307 OPT_CALLBACK_F('s', "hash", &show_one_opts
, N_("n"),
308 N_("only show SHA1 hash using <n> digits"),
309 PARSE_OPT_OPTARG
, &hash_callback
),
310 OPT__ABBREV(&show_one_opts
.abbrev
),
311 OPT__QUIET(&show_one_opts
.quiet
,
312 N_("do not print results to stdout (useful with --verify)")),
313 OPT_CALLBACK_F(0, "exclude-existing", &exclude_existing_opts
,
314 N_("pattern"), N_("show refs from stdin that aren't in local repository"),
315 PARSE_OPT_OPTARG
| PARSE_OPT_NONEG
, exclude_existing_callback
),
319 git_config(git_default_config
, NULL
);
321 argc
= parse_options(argc
, argv
, prefix
, show_ref_options
,
324 die_for_incompatible_opt3(exclude_existing_opts
.enabled
, "--exclude-existing",
328 if (exclude_existing_opts
.enabled
)
329 return cmd_show_ref__exclude_existing(&exclude_existing_opts
);
331 return cmd_show_ref__verify(&show_one_opts
, argv
);
333 return cmd_show_ref__exists(argv
);
335 return cmd_show_ref__patterns(&patterns_opts
, &show_one_opts
, argv
);