git-branch, git-checkout: autosetup for remote branch tracking
[gitweb.git] / builtin-bundle.c
index 521bbdad8b4f0f68527128e8391188926d58b661..279b8f8e582665dda3c99d800094db2f5cf5af99 100644 (file)
@@ -19,7 +19,7 @@ static const char bundle_signature[] = "# v2 git bundle\n";
 
 struct ref_list {
        unsigned int nr, alloc;
-       struct {
+       struct ref_list_entry {
                unsigned char sha1[20];
                char *name;
        } *list;
@@ -167,33 +167,54 @@ static int verify_bundle(struct bundle_header *header)
         * to be verbose about the errors
         */
        struct ref_list *p = &header->prerequisites;
-       char **argv;
-       int pid, out, i, ret = 0;
-       char buffer[1024];
+       struct rev_info revs;
+       const char *argv[] = {NULL, "--all"};
+       struct object_array refs;
+       struct commit *commit;
+       int i, ret = 0, req_nr;
+       const char *message = "Repository lacks these prerequisite commits:";
 
-       argv = xmalloc((p->nr + 4) * sizeof(const char *));
-       argv[0] = "rev-list";
-       argv[1] = "--not";
-       argv[2] = "--all";
-       for (i = 0; i < p->nr; i++)
-               argv[i + 3] = xstrdup(sha1_to_hex(p->list[i].sha1));
-       argv[p->nr + 3] = NULL;
-       out = -1;
-       pid = fork_with_pipe((const char **)argv, NULL, &out);
-       if (pid < 0)
-               return error("Could not fork rev-list");
-       while (read_string(out, buffer, sizeof(buffer)) > 0)
-               ; /* do nothing */
-       close(out);
-       for (i = 0; i < p->nr; i++)
-               free(argv[i + 3]);
-       free(argv);
-
-       while (waitpid(pid, &i, 0) < 0)
-               if (errno != EINTR)
-                       return -1;
-       if (!ret && (!WIFEXITED(i) || WEXITSTATUS(i)))
-               return error("At least one prerequisite is lacking.");
+       init_revisions(&revs, NULL);
+       for (i = 0; i < p->nr; i++) {
+               struct ref_list_entry *e = p->list + i;
+               struct object *o = parse_object(e->sha1);
+               if (o) {
+                       o->flags |= BOUNDARY_SHOW;
+                       add_pending_object(&revs, o, e->name);
+                       continue;
+               }
+               if (++ret == 1)
+                       error(message);
+               error("%s %s", sha1_to_hex(e->sha1), e->name);
+       }
+       if (revs.pending.nr == 0)
+               return ret;
+       req_nr = revs.pending.nr;
+       setup_revisions(2, argv, &revs, NULL);
+
+       memset(&refs, 0, sizeof(struct object_array));
+       for (i = 0; i < revs.pending.nr; i++) {
+               struct object_array_entry *e = revs.pending.objects + i;
+               add_object_array(e->item, e->name, &refs);
+       }
+
+       prepare_revision_walk(&revs);
+
+       i = req_nr;
+       while (i && (commit = get_revision(&revs)))
+               if (commit->object.flags & BOUNDARY_SHOW)
+                       i--;
+
+       for (i = 0; i < req_nr; i++)
+               if (!(refs.objects[i].item->flags & SHOWN)) {
+                       if (++ret == 1)
+                               error(message);
+                       error("%s %s", sha1_to_hex(refs.objects[i].item->sha1),
+                               refs.objects[i].name);
+               }
+
+       for (i = 0; i < refs.nr; i++)
+               clear_commit_marks((struct commit *)refs.objects[i].item, -1);
 
        return ret;
 }
@@ -242,11 +263,16 @@ static void show_object(struct object_array_entry *p)
        write_or_die(1, "\n", 1);
 }
 
+static void show_edge(struct commit *commit)
+{
+       ; /* nothing to do */
+}
+
 static int create_bundle(struct bundle_header *header, const char *path,
                int argc, const char **argv)
 {
        int bundle_fd = -1;
-       const char **argv_boundary = xmalloc((argc + 3) * sizeof(const char *));
+       const char **argv_boundary = xmalloc((argc + 4) * sizeof(const char *));
        const char **argv_pack = xmalloc(4 * sizeof(const char *));
        int pid, in, out, i, status;
        char buffer[1024];
@@ -261,10 +287,11 @@ static int create_bundle(struct bundle_header *header, const char *path,
        write_or_die(bundle_fd, bundle_signature, strlen(bundle_signature));
 
        /* write prerequisites */
-       memcpy(argv_boundary + 2, argv + 1, argc * sizeof(const char *));
+       memcpy(argv_boundary + 3, argv + 1, argc * sizeof(const char *));
        argv_boundary[0] = "rev-list";
        argv_boundary[1] = "--boundary";
-       argv_boundary[argc + 1] = NULL;
+       argv_boundary[2] = "--pretty=oneline";
+       argv_boundary[argc + 2] = NULL;
        out = -1;
        pid = fork_with_pipe(argv_boundary, NULL, &out);
        if (pid < 0)
@@ -319,6 +346,7 @@ static int create_bundle(struct bundle_header *header, const char *path,
        dup2(in, 1);
        close(in);
        prepare_revision_walk(&revs);
+       mark_edges_uninteresting(revs.commits, &revs, show_edge);
        traverse_commit_list(&revs, show_commit, show_object);
        close(1);
        while (waitpid(pid, &status, 0) < 0)