5 #include "object-store.h"
6 #include "repository.h"
8 struct flag_definition
{
18 static unsigned int parse_flags(const char *str
, struct flag_definition
*defs
)
20 struct string_list masks
= STRING_LIST_INIT_DUP
;
22 unsigned int result
= 0;
24 if (!strcmp(str
, "0"))
27 string_list_split(&masks
, str
, ',', 64);
28 for (; i
< masks
.nr
; i
++) {
29 const char *name
= masks
.items
[i
].string
;
30 struct flag_definition
*def
= defs
;
33 if (!strcmp(def
->name
, name
)) {
41 die("unknown flag \"%s\"", name
);
44 string_list_clear(&masks
, 0);
48 static struct flag_definition empty_flags
[] = { { NULL
, 0 } };
50 static const char *notnull(const char *arg
, const char *name
)
53 die("%s required", name
);
57 static unsigned int arg_flags(const char *arg
, const char *name
,
58 struct flag_definition
*defs
)
60 return parse_flags(notnull(arg
, name
), defs
);
63 static const char **get_store(const char **argv
, struct ref_store
**refs
)
68 die("ref store required");
69 } else if (!strcmp(argv
[0], "main")) {
70 *refs
= get_main_ref_store(the_repository
);
71 } else if (skip_prefix(argv
[0], "submodule:", &gitdir
)) {
72 struct strbuf sb
= STRBUF_INIT
;
75 ret
= strbuf_git_path_submodule(&sb
, gitdir
, "objects/");
77 die("strbuf_git_path_submodule failed: %d", ret
);
78 add_to_alternates_memory(sb
.buf
);
81 *refs
= get_submodule_ref_store(gitdir
);
82 } else if (skip_prefix(argv
[0], "worktree:", &gitdir
)) {
83 struct worktree
**p
, **worktrees
= get_worktrees();
85 for (p
= worktrees
; *p
; p
++) {
86 struct worktree
*wt
= *p
;
89 /* special case for main worktree */
90 if (!strcmp(gitdir
, "main"))
92 } else if (!strcmp(gitdir
, wt
->id
))
96 die("no such worktree: %s", gitdir
);
98 *refs
= get_worktree_ref_store(*p
);
99 free_worktrees(worktrees
);
101 die("unknown backend %s", argv
[0]);
106 /* consume store-specific optional arguments if needed */
111 static struct flag_definition pack_flags
[] = { FLAG_DEF(PACK_REFS_PRUNE
),
112 FLAG_DEF(PACK_REFS_ALL
),
115 static int cmd_pack_refs(struct ref_store
*refs
, const char **argv
)
117 unsigned int flags
= arg_flags(*argv
++, "flags", pack_flags
);
119 return refs_pack_refs(refs
, flags
);
122 static int cmd_create_symref(struct ref_store
*refs
, const char **argv
)
124 const char *refname
= notnull(*argv
++, "refname");
125 const char *target
= notnull(*argv
++, "target");
126 const char *logmsg
= *argv
++;
128 return refs_create_symref(refs
, refname
, target
, logmsg
);
131 static struct flag_definition transaction_flags
[] = {
132 FLAG_DEF(REF_NO_DEREF
),
133 FLAG_DEF(REF_FORCE_CREATE_REFLOG
),
134 FLAG_DEF(REF_SKIP_OID_VERIFICATION
),
135 FLAG_DEF(REF_SKIP_REFNAME_VERIFICATION
),
139 static int cmd_delete_refs(struct ref_store
*refs
, const char **argv
)
141 unsigned int flags
= arg_flags(*argv
++, "flags", transaction_flags
);
142 const char *msg
= *argv
++;
143 struct string_list refnames
= STRING_LIST_INIT_NODUP
;
147 string_list_append(&refnames
, *argv
++);
149 result
= refs_delete_refs(refs
, msg
, &refnames
, flags
);
150 string_list_clear(&refnames
, 0);
154 static int cmd_rename_ref(struct ref_store
*refs
, const char **argv
)
156 const char *oldref
= notnull(*argv
++, "oldref");
157 const char *newref
= notnull(*argv
++, "newref");
158 const char *logmsg
= *argv
++;
160 return refs_rename_ref(refs
, oldref
, newref
, logmsg
);
163 static int each_ref(const char *refname
, const struct object_id
*oid
,
164 int flags
, void *cb_data UNUSED
)
166 printf("%s %s 0x%x\n", oid_to_hex(oid
), refname
, flags
);
170 static int cmd_for_each_ref(struct ref_store
*refs
, const char **argv
)
172 const char *prefix
= notnull(*argv
++, "prefix");
174 return refs_for_each_ref_in(refs
, prefix
, each_ref
, NULL
);
177 static int cmd_resolve_ref(struct ref_store
*refs
, const char **argv
)
179 struct object_id oid
= *null_oid();
180 const char *refname
= notnull(*argv
++, "refname");
181 int resolve_flags
= arg_flags(*argv
++, "resolve-flags", empty_flags
);
185 ref
= refs_resolve_ref_unsafe(refs
, refname
, resolve_flags
,
187 printf("%s %s 0x%x\n", oid_to_hex(&oid
), ref
? ref
: "(null)", flags
);
191 static int cmd_verify_ref(struct ref_store
*refs
, const char **argv
)
193 const char *refname
= notnull(*argv
++, "refname");
194 struct strbuf err
= STRBUF_INIT
;
197 ret
= refs_verify_refname_available(refs
, refname
, NULL
, NULL
, &err
);
203 static int cmd_for_each_reflog(struct ref_store
*refs
, const char **argv
)
205 return refs_for_each_reflog(refs
, each_ref
, NULL
);
208 static int each_reflog(struct object_id
*old_oid
, struct object_id
*new_oid
,
209 const char *committer
, timestamp_t timestamp
,
210 int tz
, const char *msg
, void *cb_data UNUSED
)
212 printf("%s %s %s %" PRItime
" %+05d%s%s", oid_to_hex(old_oid
),
213 oid_to_hex(new_oid
), committer
, timestamp
, tz
,
214 *msg
== '\n' ? "" : "\t", msg
);
218 static int cmd_for_each_reflog_ent(struct ref_store
*refs
, const char **argv
)
220 const char *refname
= notnull(*argv
++, "refname");
222 return refs_for_each_reflog_ent(refs
, refname
, each_reflog
, refs
);
225 static int cmd_for_each_reflog_ent_reverse(struct ref_store
*refs
, const char **argv
)
227 const char *refname
= notnull(*argv
++, "refname");
229 return refs_for_each_reflog_ent_reverse(refs
, refname
, each_reflog
, refs
);
232 static int cmd_reflog_exists(struct ref_store
*refs
, const char **argv
)
234 const char *refname
= notnull(*argv
++, "refname");
236 return !refs_reflog_exists(refs
, refname
);
239 static int cmd_create_reflog(struct ref_store
*refs
, const char **argv
)
241 const char *refname
= notnull(*argv
++, "refname");
242 struct strbuf err
= STRBUF_INIT
;
245 ret
= refs_create_reflog(refs
, refname
, &err
);
251 static int cmd_delete_reflog(struct ref_store
*refs
, const char **argv
)
253 const char *refname
= notnull(*argv
++, "refname");
255 return refs_delete_reflog(refs
, refname
);
258 static int cmd_reflog_expire(struct ref_store
*refs
, const char **argv
)
260 die("not supported yet");
263 static int cmd_delete_ref(struct ref_store
*refs
, const char **argv
)
265 const char *msg
= notnull(*argv
++, "msg");
266 const char *refname
= notnull(*argv
++, "refname");
267 const char *sha1_buf
= notnull(*argv
++, "old-sha1");
268 unsigned int flags
= arg_flags(*argv
++, "flags", transaction_flags
);
269 struct object_id old_oid
;
271 if (get_oid_hex(sha1_buf
, &old_oid
))
272 die("cannot parse %s as %s", sha1_buf
, the_hash_algo
->name
);
274 return refs_delete_ref(refs
, msg
, refname
, &old_oid
, flags
);
277 static int cmd_update_ref(struct ref_store
*refs
, const char **argv
)
279 const char *msg
= notnull(*argv
++, "msg");
280 const char *refname
= notnull(*argv
++, "refname");
281 const char *new_sha1_buf
= notnull(*argv
++, "new-sha1");
282 const char *old_sha1_buf
= notnull(*argv
++, "old-sha1");
283 unsigned int flags
= arg_flags(*argv
++, "flags", transaction_flags
);
284 struct object_id old_oid
;
285 struct object_id new_oid
;
287 if (get_oid_hex(old_sha1_buf
, &old_oid
))
288 die("cannot parse %s as %s", old_sha1_buf
, the_hash_algo
->name
);
289 if (get_oid_hex(new_sha1_buf
, &new_oid
))
290 die("cannot parse %s as %s", new_sha1_buf
, the_hash_algo
->name
);
292 return refs_update_ref(refs
, msg
, refname
,
294 flags
, UPDATE_REFS_DIE_ON_ERR
);
299 int (*func
)(struct ref_store
*refs
, const char **argv
);
302 static struct command commands
[] = {
303 { "pack-refs", cmd_pack_refs
},
304 { "create-symref", cmd_create_symref
},
305 { "delete-refs", cmd_delete_refs
},
306 { "rename-ref", cmd_rename_ref
},
307 { "for-each-ref", cmd_for_each_ref
},
308 { "resolve-ref", cmd_resolve_ref
},
309 { "verify-ref", cmd_verify_ref
},
310 { "for-each-reflog", cmd_for_each_reflog
},
311 { "for-each-reflog-ent", cmd_for_each_reflog_ent
},
312 { "for-each-reflog-ent-reverse", cmd_for_each_reflog_ent_reverse
},
313 { "reflog-exists", cmd_reflog_exists
},
314 { "create-reflog", cmd_create_reflog
},
315 { "delete-reflog", cmd_delete_reflog
},
316 { "reflog-expire", cmd_reflog_expire
},
318 * backend transaction functions can't be tested separately
320 { "delete-ref", cmd_delete_ref
},
321 { "update-ref", cmd_update_ref
},
325 int cmd__ref_store(int argc
, const char **argv
)
327 struct ref_store
*refs
;
331 setup_git_directory();
333 argv
= get_store(argv
+ 1, &refs
);
337 die("ref function required");
338 for (cmd
= commands
; cmd
->name
; cmd
++) {
339 if (!strcmp(func
, cmd
->name
))
340 return cmd
->func(refs
, argv
);
342 die("unknown function %s", func
);