1#include "test-tool.h"
   2#include "cache.h"
   3#include "refs.h"
   4#include "worktree.h"
   5#include "object-store.h"
   6static const char *notnull(const char *arg, const char *name)
   8{
   9        if (!arg)
  10                die("%s required", name);
  11        return arg;
  12}
  13static unsigned int arg_flags(const char *arg, const char *name)
  15{
  16        return atoi(notnull(arg, name));
  17}
  18static const char **get_store(const char **argv, struct ref_store **refs)
  20{
  21        const char *gitdir;
  22        if (!argv[0]) {
  24                die("ref store required");
  25        } else if (!strcmp(argv[0], "main")) {
  26                *refs = get_main_ref_store();
  27        } else if (skip_prefix(argv[0], "submodule:", &gitdir)) {
  28                struct strbuf sb = STRBUF_INIT;
  29                int ret;
  30                ret = strbuf_git_path_submodule(&sb, gitdir, "objects/");
  32                if (ret)
  33                        die("strbuf_git_path_submodule failed: %d", ret);
  34                add_to_alternates_memory(sb.buf);
  35                strbuf_release(&sb);
  36                *refs = get_submodule_ref_store(gitdir);
  38        } else if (skip_prefix(argv[0], "worktree:", &gitdir)) {
  39                struct worktree **p, **worktrees = get_worktrees(0);
  40                for (p = worktrees; *p; p++) {
  42                        struct worktree *wt = *p;
  43                        if (!wt->id) {
  45                                /* special case for main worktree */
  46                                if (!strcmp(gitdir, "main"))
  47                                        break;
  48                        } else if (!strcmp(gitdir, wt->id))
  49                                break;
  50                }
  51                if (!*p)
  52                        die("no such worktree: %s", gitdir);
  53                *refs = get_worktree_ref_store(*p);
  55        } else
  56                die("unknown backend %s", argv[0]);
  57        if (!*refs)
  59                die("no ref store");
  60        /* consume store-specific optional arguments if needed */
  62        return argv + 1;
  64}
  65static int cmd_pack_refs(struct ref_store *refs, const char **argv)
  68{
  69        unsigned int flags = arg_flags(*argv++, "flags");
  70        return refs_pack_refs(refs, flags);
  72}
  73static int cmd_peel_ref(struct ref_store *refs, const char **argv)
  75{
  76        const char *refname = notnull(*argv++, "refname");
  77        struct object_id oid;
  78        int ret;
  79        ret = refs_peel_ref(refs, refname, &oid);
  81        if (!ret)
  82                puts(oid_to_hex(&oid));
  83        return ret;
  84}
  85static int cmd_create_symref(struct ref_store *refs, const char **argv)
  87{
  88        const char *refname = notnull(*argv++, "refname");
  89        const char *target = notnull(*argv++, "target");
  90        const char *logmsg = *argv++;
  91        return refs_create_symref(refs, refname, target, logmsg);
  93}
  94static int cmd_delete_refs(struct ref_store *refs, const char **argv)
  96{
  97        unsigned int flags = arg_flags(*argv++, "flags");
  98        const char *msg = *argv++;
  99        struct string_list refnames = STRING_LIST_INIT_NODUP;
 100        while (*argv)
 102                string_list_append(&refnames, *argv++);
 103        return refs_delete_refs(refs, msg, &refnames, flags);
 105}
 106static int cmd_rename_ref(struct ref_store *refs, const char **argv)
 108{
 109        const char *oldref = notnull(*argv++, "oldref");
 110        const char *newref = notnull(*argv++, "newref");
 111        const char *logmsg = *argv++;
 112        return refs_rename_ref(refs, oldref, newref, logmsg);
 114}
 115static int each_ref(const char *refname, const struct object_id *oid,
 117                    int flags, void *cb_data)
 118{
 119        printf("%s %s 0x%x\n", oid_to_hex(oid), refname, flags);
 120        return 0;
 121}
 122static int cmd_for_each_ref(struct ref_store *refs, const char **argv)
 124{
 125        const char *prefix = notnull(*argv++, "prefix");
 126        return refs_for_each_ref_in(refs, prefix, each_ref, NULL);
 128}
 129static int cmd_resolve_ref(struct ref_store *refs, const char **argv)
 131{
 132        struct object_id oid;
 133        const char *refname = notnull(*argv++, "refname");
 134        int resolve_flags = arg_flags(*argv++, "resolve-flags");
 135        int flags;
 136        const char *ref;
 137        ref = refs_resolve_ref_unsafe(refs, refname, resolve_flags,
 139                                      &oid, &flags);
 140        printf("%s %s 0x%x\n", oid_to_hex(&oid), ref ? ref : "(null)", flags);
 141        return ref ? 0 : 1;
 142}
 143static int cmd_verify_ref(struct ref_store *refs, const char **argv)
 145{
 146        const char *refname = notnull(*argv++, "refname");
 147        struct strbuf err = STRBUF_INIT;
 148        int ret;
 149        ret = refs_verify_refname_available(refs, refname, NULL, NULL, &err);
 151        if (err.len)
 152                puts(err.buf);
 153        return ret;
 154}
 155static int cmd_for_each_reflog(struct ref_store *refs, const char **argv)
 157{
 158        return refs_for_each_reflog(refs, each_ref, NULL);
 159}
 160static int each_reflog(struct object_id *old_oid, struct object_id *new_oid,
 162                       const char *committer, timestamp_t timestamp,
 163                       int tz, const char *msg, void *cb_data)
 164{
 165        printf("%s %s %s %"PRItime" %d %s\n",
 166               oid_to_hex(old_oid), oid_to_hex(new_oid),
 167               committer, timestamp, tz, msg);
 168        return 0;
 169}
 170static int cmd_for_each_reflog_ent(struct ref_store *refs, const char **argv)
 172{
 173        const char *refname = notnull(*argv++, "refname");
 174        return refs_for_each_reflog_ent(refs, refname, each_reflog, refs);
 176}
 177static int cmd_for_each_reflog_ent_reverse(struct ref_store *refs, const char **argv)
 179{
 180        const char *refname = notnull(*argv++, "refname");
 181        return refs_for_each_reflog_ent_reverse(refs, refname, each_reflog, refs);
 183}
 184static int cmd_reflog_exists(struct ref_store *refs, const char **argv)
 186{
 187        const char *refname = notnull(*argv++, "refname");
 188        return !refs_reflog_exists(refs, refname);
 190}
 191static int cmd_create_reflog(struct ref_store *refs, const char **argv)
 193{
 194        const char *refname = notnull(*argv++, "refname");
 195        int force_create = arg_flags(*argv++, "force-create");
 196        struct strbuf err = STRBUF_INIT;
 197        int ret;
 198        ret = refs_create_reflog(refs, refname, force_create, &err);
 200        if (err.len)
 201                puts(err.buf);
 202        return ret;
 203}
 204static int cmd_delete_reflog(struct ref_store *refs, const char **argv)
 206{
 207        const char *refname = notnull(*argv++, "refname");
 208        return refs_delete_reflog(refs, refname);
 210}
 211static int cmd_reflog_expire(struct ref_store *refs, const char **argv)
 213{
 214        die("not supported yet");
 215}
 216static int cmd_delete_ref(struct ref_store *refs, const char **argv)
 218{
 219        const char *msg = notnull(*argv++, "msg");
 220        const char *refname = notnull(*argv++, "refname");
 221        const char *sha1_buf = notnull(*argv++, "old-sha1");
 222        unsigned int flags = arg_flags(*argv++, "flags");
 223        struct object_id old_oid;
 224        if (get_oid_hex(sha1_buf, &old_oid))
 226                die("not sha-1");
 227        return refs_delete_ref(refs, msg, refname, &old_oid, flags);
 229}
 230static int cmd_update_ref(struct ref_store *refs, const char **argv)
 232{
 233        const char *msg = notnull(*argv++, "msg");
 234        const char *refname = notnull(*argv++, "refname");
 235        const char *new_sha1_buf = notnull(*argv++, "old-sha1");
 236        const char *old_sha1_buf = notnull(*argv++, "old-sha1");
 237        unsigned int flags = arg_flags(*argv++, "flags");
 238        struct object_id old_oid;
 239        struct object_id new_oid;
 240        if (get_oid_hex(old_sha1_buf, &old_oid) ||
 242            get_oid_hex(new_sha1_buf, &new_oid))
 243                die("not sha-1");
 244        return refs_update_ref(refs, msg, refname,
 246                               &new_oid, &old_oid,
 247                               flags, UPDATE_REFS_DIE_ON_ERR);
 248}
 249struct command {
 251        const char *name;
 252        int (*func)(struct ref_store *refs, const char **argv);
 253};
 254static struct command commands[] = {
 256        { "pack-refs", cmd_pack_refs },
 257        { "peel-ref", cmd_peel_ref },
 258        { "create-symref", cmd_create_symref },
 259        { "delete-refs", cmd_delete_refs },
 260        { "rename-ref", cmd_rename_ref },
 261        { "for-each-ref", cmd_for_each_ref },
 262        { "resolve-ref", cmd_resolve_ref },
 263        { "verify-ref", cmd_verify_ref },
 264        { "for-each-reflog", cmd_for_each_reflog },
 265        { "for-each-reflog-ent", cmd_for_each_reflog_ent },
 266        { "for-each-reflog-ent-reverse", cmd_for_each_reflog_ent_reverse },
 267        { "reflog-exists", cmd_reflog_exists },
 268        { "create-reflog", cmd_create_reflog },
 269        { "delete-reflog", cmd_delete_reflog },
 270        { "reflog-expire", cmd_reflog_expire },
 271        /*
 272         * backend transaction functions can't be tested separately
 273         */
 274        { "delete-ref", cmd_delete_ref },
 275        { "update-ref", cmd_update_ref },
 276        { NULL, NULL }
 277};
 278int cmd__ref_store(int argc, const char **argv)
 280{
 281        struct ref_store *refs;
 282        const char *func;
 283        struct command *cmd;
 284        setup_git_directory();
 286        argv = get_store(argv + 1, &refs);
 288        func = *argv++;
 290        if (!func)
 291                die("ref function required");
 292        for (cmd = commands; cmd->name; cmd++) {
 293                if (!strcmp(func, cmd->name))
 294                        return cmd->func(refs, argv);
 295        }
 296        die("unknown function %s", func);
 297        return 0;
 298}