| #include "test-lib.h" |
| #include "lib-oid.h" |
| #include "oidtree.h" |
| #include "hash.h" |
| #include "hex.h" |
| #include "strvec.h" |
| |
| #define FILL_TREE(tree, ...) \ |
| do { \ |
| const char *hexes[] = { __VA_ARGS__ }; \ |
| if (fill_tree_loc(tree, hexes, ARRAY_SIZE(hexes))) \ |
| return; \ |
| } while (0) |
| |
| static int fill_tree_loc(struct oidtree *ot, const char *hexes[], size_t n) |
| { |
| for (size_t i = 0; i < n; i++) { |
| struct object_id oid; |
| if (!check_int(get_oid_arbitrary_hex(hexes[i], &oid), ==, 0)) |
| return -1; |
| oidtree_insert(ot, &oid); |
| } |
| return 0; |
| } |
| |
| static void check_contains(struct oidtree *ot, const char *hex, int expected) |
| { |
| struct object_id oid; |
| |
| if (!check_int(get_oid_arbitrary_hex(hex, &oid), ==, 0)) |
| return; |
| if (!check_int(oidtree_contains(ot, &oid), ==, expected)) |
| test_msg("oid: %s", oid_to_hex(&oid)); |
| } |
| |
| struct expected_hex_iter { |
| size_t i; |
| struct strvec expected_hexes; |
| const char *query; |
| }; |
| |
| static enum cb_next check_each_cb(const struct object_id *oid, void *data) |
| { |
| struct expected_hex_iter *hex_iter = data; |
| struct object_id expected; |
| |
| if (!check_int(hex_iter->i, <, hex_iter->expected_hexes.nr)) { |
| test_msg("error: extraneous callback for query: ('%s'), object_id: ('%s')", |
| hex_iter->query, oid_to_hex(oid)); |
| return CB_BREAK; |
| } |
| |
| if (!check_int(get_oid_arbitrary_hex(hex_iter->expected_hexes.v[hex_iter->i], |
| &expected), ==, 0)) |
| ; /* the data is bogus and cannot be used */ |
| else if (!check(oideq(oid, &expected))) |
| test_msg("expected: %s\n got: %s\n query: %s", |
| oid_to_hex(&expected), oid_to_hex(oid), hex_iter->query); |
| |
| hex_iter->i += 1; |
| return CB_CONTINUE; |
| } |
| |
| LAST_ARG_MUST_BE_NULL |
| static void check_each(struct oidtree *ot, const char *query, ...) |
| { |
| struct object_id oid; |
| struct expected_hex_iter hex_iter = { .expected_hexes = STRVEC_INIT, |
| .query = query }; |
| const char *arg; |
| va_list hex_args; |
| |
| va_start(hex_args, query); |
| while ((arg = va_arg(hex_args, const char *))) |
| strvec_push(&hex_iter.expected_hexes, arg); |
| va_end(hex_args); |
| |
| if (!check_int(get_oid_arbitrary_hex(query, &oid), ==, 0)) |
| return; |
| oidtree_each(ot, &oid, strlen(query), check_each_cb, &hex_iter); |
| |
| if (!check_int(hex_iter.i, ==, hex_iter.expected_hexes.nr)) |
| test_msg("error: could not find some 'object_id's for query ('%s')", query); |
| strvec_clear(&hex_iter.expected_hexes); |
| } |
| |
| static void setup(void (*f)(struct oidtree *ot)) |
| { |
| struct oidtree ot; |
| |
| oidtree_init(&ot); |
| f(&ot); |
| oidtree_clear(&ot); |
| } |
| |
| static void t_contains(struct oidtree *ot) |
| { |
| FILL_TREE(ot, "444", "1", "2", "3", "4", "5", "a", "b", "c", "d", "e"); |
| check_contains(ot, "44", 0); |
| check_contains(ot, "441", 0); |
| check_contains(ot, "440", 0); |
| check_contains(ot, "444", 1); |
| check_contains(ot, "4440", 1); |
| check_contains(ot, "4444", 0); |
| } |
| |
| static void t_each(struct oidtree *ot) |
| { |
| FILL_TREE(ot, "f", "9", "8", "123", "321", "320", "a", "b", "c", "d", "e"); |
| check_each(ot, "12300", "123", NULL); |
| check_each(ot, "3211", NULL); /* should not reach callback */ |
| check_each(ot, "3210", "321", NULL); |
| check_each(ot, "32100", "321", NULL); |
| check_each(ot, "32", "320", "321", NULL); |
| } |
| |
| int cmd_main(int argc UNUSED, const char **argv UNUSED) |
| { |
| TEST(setup(t_contains), "oidtree insert and contains works"); |
| TEST(setup(t_each), "oidtree each works"); |
| return test_done(); |
| } |