Message ID | b7e9b81e8b32313f00d38257ba731e73d17224cb.1733514358.git.gitgitgadget@gmail.com (mailing list archive) |
---|---|
State | New |
Headers | show |
Series | PATH WALK I: The path-walk API | expand |
On Fri, Dec 06, 2024 at 07:45:52PM +0000, Derrick Stolee via GitGitGadget wrote: > --- /dev/null > +++ b/path-walk.c > @@ -0,0 +1,263 @@ > +/* > + * path-walk.c: implementation for path-based walks of the object graph. > + */ > +#include "git-compat-util.h" > +#include "path-walk.h" > +#include "blob.h" > +#include "commit.h" > +#include "dir.h" > +#include "hashmap.h" > +#include "hex.h" > +#include "object.h" > +#include "oid-array.h" > +#include "revision.h" > +#include "string-list.h" > +#include "strmap.h" > +#include "trace2.h" > +#include "tree.h" > +#include "tree-walk.h" > + > +struct type_and_oid_list > +{ > + enum object_type type; > + struct oid_array oids; > +}; Nit: formatting of this struct is off. > +static void push_to_stack(struct path_walk_context *ctx, > + const char *path) > +{ > + if (strset_contains(&ctx->path_stack_pushed, path)) > + return; > + > + strset_add(&ctx->path_stack_pushed, path); > + string_list_append(&ctx->path_stack, path); > +} > + > +static int add_children(struct path_walk_context *ctx, > + const char *base_path, > + struct object_id *oid) > +{ So this function assumes that `oid` always refers to a tree? I think it would make sense to clarify this by calling the function accordingly, like e.g. `add_tree_entries()`. > + struct tree_desc desc; > + struct name_entry entry; > + struct strbuf path = STRBUF_INIT; > + size_t base_len; > + struct tree *tree = lookup_tree(ctx->repo, oid); > + > + if (!tree) { > + error(_("failed to walk children of tree %s: not found"), > + oid_to_hex(oid)); > + return -1; > + } else if (parse_tree_gently(tree, 1)) { > + die("bad tree object %s", oid_to_hex(oid)); I wonder whether we maybe shouldn't die but instead return an error in the spirit of libification. > + } > + strbuf_addstr(&path, base_path); > + base_len = path.len; > + > + parse_tree(tree); > + init_tree_desc(&desc, &tree->object.oid, tree->buffer, tree->size); > + while (tree_entry(&desc, &entry)) { > + struct type_and_oid_list *list; > + struct object *o; > + /* Not actually true, but we will ignore submodules later. */ > + enum object_type type = S_ISDIR(entry.mode) ? OBJ_TREE : OBJ_BLOB; > + > + /* Skip submodules. */ > + if (S_ISGITLINK(entry.mode)) > + continue; > + > + if (type == OBJ_TREE) { > + struct tree *child = lookup_tree(ctx->repo, &entry.oid); > + o = child ? &child->object : NULL; > + } else if (type == OBJ_BLOB) { > + struct blob *child = lookup_blob(ctx->repo, &entry.oid); > + o = child ? &child->object : NULL; > + } else { > + /* Wrong type? */ > + continue; This code is unreachable, so we could make this a `BUG()`. Might also use a switch instead, but that's more of a stylistic question. > + } > + > + if (!o) /* report error?*/ > + continue; So this can only happen in case `lookup_tree()` or `lookup_blob()` run into an error. I think this error should definitely be bubbled up so that we don't silently skip tree entries in case of repo corruption. > + strbuf_setlen(&path, base_len); > + strbuf_add(&path, entry.path, entry.pathlen); > + > + /* > + * Trees will end with "/" for concatenation and distinction > + * from blobs at the same path. > + */ > + if (type == OBJ_TREE) > + strbuf_addch(&path, '/'); > + > + if (!(list = strmap_get(&ctx->paths_to_lists, path.buf))) { > + CALLOC_ARRAY(list, 1); > + list->type = type; > + strmap_put(&ctx->paths_to_lists, path.buf, list); > + } > + push_to_stack(ctx, path.buf); > + > + /* Skip this object if already seen. */ > + if (o->flags & SEEN) > + continue; > + o->flags |= SEEN; This made me wonder: why do we only skip the object this late? Couldn't we already have done so immediately after we have looked up the object to avoid some work? If not, it might be useful to add a comment why it has to come this late. > + oid_array_append(&list->oids, &entry.oid); > + } > + > + free_tree_buffer(tree); > + strbuf_release(&path); > + return 0; > +} > + > +/* > + * For each path in paths_to_explore, walk the trees another level > + * and add any found blobs to the batch (but only if they exist and > + * haven't been added yet). > + */ > +static int walk_path(struct path_walk_context *ctx, > + const char *path) > +{ > + struct type_and_oid_list *list; > + int ret = 0; Nit: needless initialization. > + > + list = strmap_get(&ctx->paths_to_lists, path); > + > + if (!list->oids.nr) > + return 0; > + > + /* Evaluate function pointer on this data. */ > + ret = ctx->info->path_fn(path, &list->oids, list->type, > + ctx->info->path_fn_data); > + > + /* Expand data for children. */ > + if (list->type == OBJ_TREE) { > + for (size_t i = 0; i < list->oids.nr; i++) { > + ret |= add_children(ctx, > + path, > + &list->oids.oid[i]); > + } > + } > + > + oid_array_clear(&list->oids); > + strmap_remove(&ctx->paths_to_lists, path, 1); > + return ret; > +} > + > +static void clear_strmap(struct strmap *map) Nit: this isn't clearing a generic strmap, but rather `paths_to_lists`. Should we maybe rename it to `clear_paths_to_lists()`? > +{ > + struct hashmap_iter iter; > + struct strmap_entry *e; > + > + hashmap_for_each_entry(&map->map, &iter, e, ent) { > + struct type_and_oid_list *list = e->value; > + oid_array_clear(&list->oids); > + } > + strmap_clear(map, 1); > + strmap_init(map); > +} > + > +/** > + * Given the configuration of 'info', walk the commits based on 'info->revs' and > + * call 'info->path_fn' on each discovered path. > + * > + * Returns nonzero on an error. > + */ > +int walk_objects_by_path(struct path_walk_info *info) > +{ > + const char *root_path = ""; > + int ret = 0; > + size_t commits_nr = 0, paths_nr = 0; > + struct commit *c; > + struct type_and_oid_list *root_tree_list; > + struct path_walk_context ctx = { > + .repo = info->revs->repo, > + .revs = info->revs, > + .info = info, > + .path_stack = STRING_LIST_INIT_DUP, > + .path_stack_pushed = STRSET_INIT, > + .paths_to_lists = STRMAP_INIT > + }; > + > + trace2_region_enter("path-walk", "commit-walk", info->revs->repo); > + > + /* Insert a single list for the root tree into the paths. */ > + CALLOC_ARRAY(root_tree_list, 1); > + root_tree_list->type = OBJ_TREE; > + strmap_put(&ctx.paths_to_lists, root_path, root_tree_list); > + push_to_stack(&ctx, root_path); > + > + if (prepare_revision_walk(info->revs)) > + die(_("failed to setup revision walk")); > + > + while ((c = get_revision(info->revs))) { > + struct object_id *oid = get_commit_tree_oid(c); > + struct tree *t; > + commits_nr++; > + > + oid = get_commit_tree_oid(c); > + t = lookup_tree(info->revs->repo, oid); > + > + if (!t) { > + warning("could not find tree %s", oid_to_hex(oid)); > + continue; > + } Is this error something we should bubble up to the caller? As mentioned above, I'm being cautious about silently accepting potentially-corrupt data. Silent in the spirit of the caller not noticing it, not in the sense of the user not noticing it. Patrick
On 12/13/24 6:58 AM, Patrick Steinhardt wrote: > On Fri, Dec 06, 2024 at 07:45:52PM +0000, Derrick Stolee via GitGitGadget wrote: >> + } else if (parse_tree_gently(tree, 1)) { >> + die("bad tree object %s", oid_to_hex(oid)); > > I wonder whether we maybe shouldn't die but instead return an error in > the spirit of libification. This is in fact something that is being tested when 'git pack-objects' has the --path-walk feature. See "get an error for missing tree object" in t5317 as an example. It's not enough to fail, but we need to fail with this error message. Has there been enough progress in the libification effort to establish a pattern for returning an error message like "bad tree object %s" from an API like this to the caller? I will try using a "error(); return -1;" and consider that as the best option for right now. >> + } else { >> + /* Wrong type? */ >> + continue; > > This code is unreachable, so we could make this a `BUG()`. Might also > use a switch instead, but that's more of a stylistic question. I think a BUG() would be good here. >> + } >> + >> + if (!o) /* report error?*/ >> + continue; > > So this can only happen in case `lookup_tree()` or `lookup_blob()` > run into an error. I think this error should definitely be bubbled up so > that we don't silently skip tree entries in case of repo corruption. Looks like I agreed with you but didn't follow through. >> + /* Skip this object if already seen. */ >> + if (o->flags & SEEN) >> + continue; >> + o->flags |= SEEN; > > This made me wonder: why do we only skip the object this late? Couldn't > we already have done so immediately after we have looked up the object > to avoid some work? If not, it might be useful to add a comment why it > has to come this late. I went to look to see if there was a reason, and at this point there is not a good reason. This should be moved up to avoid some checks and path manipulation. I think that in a later patch, the use of the UNINTERESTING flag is important to pass the flag even when the object is already SEEN. This is probably cruft from an earlier version that passed the UNINTERESTING bits in this part of the code. >> + if (!t) { >> + warning("could not find tree %s", oid_to_hex(oid)); >> + continue; >> + } > > Is this error something we should bubble up to the caller? As mentioned > above, I'm being cautious about silently accepting potentially-corrupt > data. Silent in the spirit of the caller not noticing it, not in the > sense of the user not noticing it. Can do. Thanks for the careful review! -Stolee
diff --git a/Documentation/technical/api-path-walk.txt b/Documentation/technical/api-path-walk.txt new file mode 100644 index 00000000000..c550c77ca30 --- /dev/null +++ b/Documentation/technical/api-path-walk.txt @@ -0,0 +1,45 @@ +Path-Walk API +============= + +The path-walk API is used to walk reachable objects, but to visit objects +in batches based on a common path they appear in, or by type. + +For example, all reachable commits are visited in a group. All tags are +visited in a group. Then, all root trees are visited. At some point, all +blobs reachable via a path `my/dir/to/A` are visited. When there are +multiple paths possible to reach the same object, then only one of those +paths is used to visit the object. + +Basics +------ + +To use the path-walk API, include `path-walk.h` and call +`walk_objects_by_path()` with a customized `path_walk_info` struct. The +struct is used to set all of the options for how the walk should proceed. +Let's dig into the different options and their use. + +`path_fn` and `path_fn_data`:: + The most important option is the `path_fn` option, which is a + function pointer to the callback that can execute logic on the + object IDs for objects grouped by type and path. This function + also receives a `data` value that corresponds to the + `path_fn_data` member, for providing custom data structures to + this callback function. + +`revs`:: + To configure the exact details of the reachable set of objects, + use the `revs` member and initialize it using the revision + machinery in `revision.h`. Initialize `revs` using calls such as + `setup_revisions()` or `parse_revision_opt()`. Do not call + `prepare_revision_walk()`, as that will be called within + `walk_objects_by_path()`. ++ +It is also important that you do not specify the `--objects` flag for the +`revs` struct. The revision walk should only be used to walk commits, and +the objects will be walked in a separate way based on those starting +commits. + +Examples +-------- + +See example usages in future changes. diff --git a/Makefile b/Makefile index 7344a7f7257..d0d8d6888e3 100644 --- a/Makefile +++ b/Makefile @@ -1094,6 +1094,7 @@ LIB_OBJS += parse-options.o LIB_OBJS += patch-delta.o LIB_OBJS += patch-ids.o LIB_OBJS += path.o +LIB_OBJS += path-walk.o LIB_OBJS += pathspec.o LIB_OBJS += pkt-line.o LIB_OBJS += preload-index.o diff --git a/path-walk.c b/path-walk.c new file mode 100644 index 00000000000..24cf04c1e7d --- /dev/null +++ b/path-walk.c @@ -0,0 +1,263 @@ +/* + * path-walk.c: implementation for path-based walks of the object graph. + */ +#include "git-compat-util.h" +#include "path-walk.h" +#include "blob.h" +#include "commit.h" +#include "dir.h" +#include "hashmap.h" +#include "hex.h" +#include "object.h" +#include "oid-array.h" +#include "revision.h" +#include "string-list.h" +#include "strmap.h" +#include "trace2.h" +#include "tree.h" +#include "tree-walk.h" + +struct type_and_oid_list +{ + enum object_type type; + struct oid_array oids; +}; + +#define TYPE_AND_OID_LIST_INIT { \ + .type = OBJ_NONE, \ + .oids = OID_ARRAY_INIT \ +} + +struct path_walk_context { + /** + * Repeats of data in 'struct path_walk_info' for + * access with fewer characters. + */ + struct repository *repo; + struct rev_info *revs; + struct path_walk_info *info; + + /** + * Map a path to a 'struct type_and_oid_list' + * containing the objects discovered at that + * path. + */ + struct strmap paths_to_lists; + + /** + * Store the current list of paths in a stack, to + * facilitate depth-first-search without recursion. + * + * Use path_stack_pushed to indicate whether a path + * was previously added to path_stack. + */ + struct string_list path_stack; + struct strset path_stack_pushed; +}; + +static void push_to_stack(struct path_walk_context *ctx, + const char *path) +{ + if (strset_contains(&ctx->path_stack_pushed, path)) + return; + + strset_add(&ctx->path_stack_pushed, path); + string_list_append(&ctx->path_stack, path); +} + +static int add_children(struct path_walk_context *ctx, + const char *base_path, + struct object_id *oid) +{ + struct tree_desc desc; + struct name_entry entry; + struct strbuf path = STRBUF_INIT; + size_t base_len; + struct tree *tree = lookup_tree(ctx->repo, oid); + + if (!tree) { + error(_("failed to walk children of tree %s: not found"), + oid_to_hex(oid)); + return -1; + } else if (parse_tree_gently(tree, 1)) { + die("bad tree object %s", oid_to_hex(oid)); + } + + strbuf_addstr(&path, base_path); + base_len = path.len; + + parse_tree(tree); + init_tree_desc(&desc, &tree->object.oid, tree->buffer, tree->size); + while (tree_entry(&desc, &entry)) { + struct type_and_oid_list *list; + struct object *o; + /* Not actually true, but we will ignore submodules later. */ + enum object_type type = S_ISDIR(entry.mode) ? OBJ_TREE : OBJ_BLOB; + + /* Skip submodules. */ + if (S_ISGITLINK(entry.mode)) + continue; + + if (type == OBJ_TREE) { + struct tree *child = lookup_tree(ctx->repo, &entry.oid); + o = child ? &child->object : NULL; + } else if (type == OBJ_BLOB) { + struct blob *child = lookup_blob(ctx->repo, &entry.oid); + o = child ? &child->object : NULL; + } else { + /* Wrong type? */ + continue; + } + + if (!o) /* report error?*/ + continue; + + strbuf_setlen(&path, base_len); + strbuf_add(&path, entry.path, entry.pathlen); + + /* + * Trees will end with "/" for concatenation and distinction + * from blobs at the same path. + */ + if (type == OBJ_TREE) + strbuf_addch(&path, '/'); + + if (!(list = strmap_get(&ctx->paths_to_lists, path.buf))) { + CALLOC_ARRAY(list, 1); + list->type = type; + strmap_put(&ctx->paths_to_lists, path.buf, list); + } + push_to_stack(ctx, path.buf); + + /* Skip this object if already seen. */ + if (o->flags & SEEN) + continue; + o->flags |= SEEN; + oid_array_append(&list->oids, &entry.oid); + } + + free_tree_buffer(tree); + strbuf_release(&path); + return 0; +} + +/* + * For each path in paths_to_explore, walk the trees another level + * and add any found blobs to the batch (but only if they exist and + * haven't been added yet). + */ +static int walk_path(struct path_walk_context *ctx, + const char *path) +{ + struct type_and_oid_list *list; + int ret = 0; + + list = strmap_get(&ctx->paths_to_lists, path); + + if (!list->oids.nr) + return 0; + + /* Evaluate function pointer on this data. */ + ret = ctx->info->path_fn(path, &list->oids, list->type, + ctx->info->path_fn_data); + + /* Expand data for children. */ + if (list->type == OBJ_TREE) { + for (size_t i = 0; i < list->oids.nr; i++) { + ret |= add_children(ctx, + path, + &list->oids.oid[i]); + } + } + + oid_array_clear(&list->oids); + strmap_remove(&ctx->paths_to_lists, path, 1); + return ret; +} + +static void clear_strmap(struct strmap *map) +{ + struct hashmap_iter iter; + struct strmap_entry *e; + + hashmap_for_each_entry(&map->map, &iter, e, ent) { + struct type_and_oid_list *list = e->value; + oid_array_clear(&list->oids); + } + strmap_clear(map, 1); + strmap_init(map); +} + +/** + * Given the configuration of 'info', walk the commits based on 'info->revs' and + * call 'info->path_fn' on each discovered path. + * + * Returns nonzero on an error. + */ +int walk_objects_by_path(struct path_walk_info *info) +{ + const char *root_path = ""; + int ret = 0; + size_t commits_nr = 0, paths_nr = 0; + struct commit *c; + struct type_and_oid_list *root_tree_list; + struct path_walk_context ctx = { + .repo = info->revs->repo, + .revs = info->revs, + .info = info, + .path_stack = STRING_LIST_INIT_DUP, + .path_stack_pushed = STRSET_INIT, + .paths_to_lists = STRMAP_INIT + }; + + trace2_region_enter("path-walk", "commit-walk", info->revs->repo); + + /* Insert a single list for the root tree into the paths. */ + CALLOC_ARRAY(root_tree_list, 1); + root_tree_list->type = OBJ_TREE; + strmap_put(&ctx.paths_to_lists, root_path, root_tree_list); + push_to_stack(&ctx, root_path); + + if (prepare_revision_walk(info->revs)) + die(_("failed to setup revision walk")); + + while ((c = get_revision(info->revs))) { + struct object_id *oid = get_commit_tree_oid(c); + struct tree *t; + commits_nr++; + + oid = get_commit_tree_oid(c); + t = lookup_tree(info->revs->repo, oid); + + if (!t) { + warning("could not find tree %s", oid_to_hex(oid)); + continue; + } + + if (t->object.flags & SEEN) + continue; + t->object.flags |= SEEN; + oid_array_append(&root_tree_list->oids, oid); + } + + trace2_data_intmax("path-walk", ctx.repo, "commits", commits_nr); + trace2_region_leave("path-walk", "commit-walk", info->revs->repo); + + trace2_region_enter("path-walk", "path-walk", info->revs->repo); + while (!ret && ctx.path_stack.nr) { + char *path = ctx.path_stack.items[ctx.path_stack.nr - 1].string; + ctx.path_stack.nr--; + paths_nr++; + + ret = walk_path(&ctx, path); + + free(path); + } + trace2_data_intmax("path-walk", ctx.repo, "paths", paths_nr); + trace2_region_leave("path-walk", "path-walk", info->revs->repo); + + clear_strmap(&ctx.paths_to_lists); + strset_clear(&ctx.path_stack_pushed); + string_list_clear(&ctx.path_stack, 0); + return ret; +} diff --git a/path-walk.h b/path-walk.h new file mode 100644 index 00000000000..c9e94a98bc8 --- /dev/null +++ b/path-walk.h @@ -0,0 +1,43 @@ +/* + * path-walk.h : Methods and structures for walking the object graph in batches + * by the paths that can reach those objects. + */ +#include "object.h" /* Required for 'enum object_type'. */ + +struct rev_info; +struct oid_array; + +/** + * The type of a function pointer for the method that is called on a list of + * objects reachable at a given path. + */ +typedef int (*path_fn)(const char *path, + struct oid_array *oids, + enum object_type type, + void *data); + +struct path_walk_info { + /** + * revs provides the definitions for the commit walk, including + * which commits are UNINTERESTING or not. + */ + struct rev_info *revs; + + /** + * The caller wishes to execute custom logic on objects reachable at a + * given path. Every reachable object will be visited exactly once, and + * the first path to see an object wins. This may not be a stable choice. + */ + path_fn path_fn; + void *path_fn_data; +}; + +#define PATH_WALK_INFO_INIT { 0 } + +/** + * Given the configuration of 'info', walk the commits based on 'info->revs' and + * call 'info->path_fn' on each discovered path. + * + * Returns nonzero on an error. + */ +int walk_objects_by_path(struct path_walk_info *info);