Re: [GSoC][PATCH] t: migrate helper/test-oidmap.c to unit-tests/t-oidmap.c

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

 



Jonathan Nieder <jrnieder@xxxxxxxxx> wrote:
> Ghanshyam Thakkar wrote:
>
> > helper/test-oidmap.c along with t0016-oidmap.sh test the oidmap.h
> > library which is built on top of hashmap.h to store arbitrary
> > datastructure (which must contain oidmap_entry, which is a wrapper
> > around object_id). These entries can be accessed by querying their
> > associated object_id.
> >
> > Migrate them to the unit testing framework for better performance,
> > concise code and better debugging. Along with the migration also plug
> > memory leaks and make the test logic independent for all the tests.
> > The migration removes 'put' tests from t0016, because it is used as
> > setup to all the other tests, so testing it separately does not yield
> > any benefit.
> >
> > Mentored-by: Christian Couder <chriscool@xxxxxxxxxxxxx>
> > Mentored-by: Kaartic Sivaraam <kaartic.sivaraam@xxxxxxxxx>
> > Signed-off-by: Ghanshyam Thakkar <shyamthakkar001@xxxxxxxxx>
> > ---
> > This patch is depenedent on 'gt/unit-test-oidtree' for lib-oid.
>
> Very neat! I'm cc-ing Josh Steadmon for unit test framework expertise.
>
> Patch left unsnipped for reference.

Friendly reminder for reviews/acks. :)

Thanks.

> >  Makefile                |   2 +-
> >  t/helper/test-oidmap.c  | 123 ------------------------------
> >  t/helper/test-tool.c    |   1 -
> >  t/helper/test-tool.h    |   1 -
> >  t/t0016-oidmap.sh       | 112 ---------------------------
> >  t/unit-tests/t-oidmap.c | 165 ++++++++++++++++++++++++++++++++++++++++
> >  6 files changed, 166 insertions(+), 238 deletions(-)
> >  delete mode 100644 t/helper/test-oidmap.c
> >  delete mode 100755 t/t0016-oidmap.sh
> >  create mode 100644 t/unit-tests/t-oidmap.c
> > 
> > diff --git a/Makefile b/Makefile
> > index 03751e0fc0..f7ed50f3a9 100644
> > --- a/Makefile
> > +++ b/Makefile
> > @@ -810,7 +810,6 @@ TEST_BUILTINS_OBJS += test-match-trees.o
> >  TEST_BUILTINS_OBJS += test-mergesort.o
> >  TEST_BUILTINS_OBJS += test-mktemp.o
> >  TEST_BUILTINS_OBJS += test-oid-array.o
> > -TEST_BUILTINS_OBJS += test-oidmap.o
> >  TEST_BUILTINS_OBJS += test-online-cpus.o
> >  TEST_BUILTINS_OBJS += test-pack-mtimes.o
> >  TEST_BUILTINS_OBJS += test-parse-options.o
> > @@ -1334,6 +1333,7 @@ THIRD_PARTY_SOURCES += sha1dc/%
> >  
> >  UNIT_TEST_PROGRAMS += t-ctype
> >  UNIT_TEST_PROGRAMS += t-mem-pool
> > +UNIT_TEST_PROGRAMS += t-oidmap
> >  UNIT_TEST_PROGRAMS += t-oidtree
> >  UNIT_TEST_PROGRAMS += t-prio-queue
> >  UNIT_TEST_PROGRAMS += t-strbuf
> > diff --git a/t/helper/test-oidmap.c b/t/helper/test-oidmap.c
> > deleted file mode 100644
> > index bd30244a54..0000000000
> > --- a/t/helper/test-oidmap.c
> > +++ /dev/null
> > @@ -1,123 +0,0 @@
> > -#include "test-tool.h"
> > -#include "hex.h"
> > -#include "object-name.h"
> > -#include "oidmap.h"
> > -#include "repository.h"
> > -#include "setup.h"
> > -#include "strbuf.h"
> > -#include "string-list.h"
> > -
> > -/* key is an oid and value is a name (could be a refname for example) */
> > -struct test_entry {
> > -	struct oidmap_entry entry;
> > -	char name[FLEX_ARRAY];
> > -};
> > -
> > -#define DELIM " \t\r\n"
> > -
> > -/*
> > - * Read stdin line by line and print result of commands to stdout:
> > - *
> > - * hash oidkey -> sha1hash(oidkey)
> > - * put oidkey namevalue -> NULL / old namevalue
> > - * get oidkey -> NULL / namevalue
> > - * remove oidkey -> NULL / old namevalue
> > - * iterate -> oidkey1 namevalue1\noidkey2 namevalue2\n...
> > - *
> > - */
> > -int cmd__oidmap(int argc UNUSED, const char **argv UNUSED)
> > -{
> > -	struct string_list parts = STRING_LIST_INIT_NODUP;
> > -	struct strbuf line = STRBUF_INIT;
> > -	struct oidmap map = OIDMAP_INIT;
> > -
> > -	setup_git_directory();
> > -
> > -	/* init oidmap */
> > -	oidmap_init(&map, 0);
> > -
> > -	/* process commands from stdin */
> > -	while (strbuf_getline(&line, stdin) != EOF) {
> > -		char *cmd, *p1, *p2;
> > -		struct test_entry *entry;
> > -		struct object_id oid;
> > -
> > -		/* break line into command and up to two parameters */
> > -		string_list_setlen(&parts, 0);
> > -		string_list_split_in_place(&parts, line.buf, DELIM, 2);
> > -		string_list_remove_empty_items(&parts, 0);
> > -
> > -		/* ignore empty lines */
> > -		if (!parts.nr)
> > -			continue;
> > -		if (!*parts.items[0].string || *parts.items[0].string == '#')
> > -			continue;
> > -
> > -		cmd = parts.items[0].string;
> > -		p1 = parts.nr >= 1 ? parts.items[1].string : NULL;
> > -		p2 = parts.nr >= 2 ? parts.items[2].string : NULL;
> > -
> > -		if (!strcmp("put", cmd) && p1 && p2) {
> > -
> > -			if (repo_get_oid(the_repository, p1, &oid)) {
> > -				printf("Unknown oid: %s\n", p1);
> > -				continue;
> > -			}
> > -
> > -			/* create entry with oid_key = p1, name_value = p2 */
> > -			FLEX_ALLOC_STR(entry, name, p2);
> > -			oidcpy(&entry->entry.oid, &oid);
> > -
> > -			/* add / replace entry */
> > -			entry = oidmap_put(&map, entry);
> > -
> > -			/* print and free replaced entry, if any */
> > -			puts(entry ? entry->name : "NULL");
> > -			free(entry);
> > -
> > -		} else if (!strcmp("get", cmd) && p1) {
> > -
> > -			if (repo_get_oid(the_repository, p1, &oid)) {
> > -				printf("Unknown oid: %s\n", p1);
> > -				continue;
> > -			}
> > -
> > -			/* lookup entry in oidmap */
> > -			entry = oidmap_get(&map, &oid);
> > -
> > -			/* print result */
> > -			puts(entry ? entry->name : "NULL");
> > -
> > -		} else if (!strcmp("remove", cmd) && p1) {
> > -
> > -			if (repo_get_oid(the_repository, p1, &oid)) {
> > -				printf("Unknown oid: %s\n", p1);
> > -				continue;
> > -			}
> > -
> > -			/* remove entry from oidmap */
> > -			entry = oidmap_remove(&map, &oid);
> > -
> > -			/* print result and free entry*/
> > -			puts(entry ? entry->name : "NULL");
> > -			free(entry);
> > -
> > -		} else if (!strcmp("iterate", cmd)) {
> > -
> > -			struct oidmap_iter iter;
> > -			oidmap_iter_init(&map, &iter);
> > -			while ((entry = oidmap_iter_next(&iter)))
> > -				printf("%s %s\n", oid_to_hex(&entry->entry.oid), entry->name);
> > -
> > -		} else {
> > -
> > -			printf("Unknown command %s\n", cmd);
> > -
> > -		}
> > -	}
> > -
> > -	string_list_clear(&parts, 0);
> > -	strbuf_release(&line);
> > -	oidmap_free(&map, 1);
> > -	return 0;
> > -}
> > diff --git a/t/helper/test-tool.c b/t/helper/test-tool.c
> > index 253324a06b..5f013d8b2b 100644
> > --- a/t/helper/test-tool.c
> > +++ b/t/helper/test-tool.c
> > @@ -45,7 +45,6 @@ static struct test_cmd cmds[] = {
> >  	{ "mergesort", cmd__mergesort },
> >  	{ "mktemp", cmd__mktemp },
> >  	{ "oid-array", cmd__oid_array },
> > -	{ "oidmap", cmd__oidmap },
> >  	{ "online-cpus", cmd__online_cpus },
> >  	{ "pack-mtimes", cmd__pack_mtimes },
> >  	{ "parse-options", cmd__parse_options },
> > diff --git a/t/helper/test-tool.h b/t/helper/test-tool.h
> > index 460dd7d260..c7d3e43694 100644
> > --- a/t/helper/test-tool.h
> > +++ b/t/helper/test-tool.h
> > @@ -38,7 +38,6 @@ int cmd__lazy_init_name_hash(int argc, const char **argv);
> >  int cmd__match_trees(int argc, const char **argv);
> >  int cmd__mergesort(int argc, const char **argv);
> >  int cmd__mktemp(int argc, const char **argv);
> > -int cmd__oidmap(int argc, const char **argv);
> >  int cmd__online_cpus(int argc, const char **argv);
> >  int cmd__pack_mtimes(int argc, const char **argv);
> >  int cmd__parse_options(int argc, const char **argv);
> > diff --git a/t/t0016-oidmap.sh b/t/t0016-oidmap.sh
> > deleted file mode 100755
> > index 0faef1f4f1..0000000000
> > --- a/t/t0016-oidmap.sh
> > +++ /dev/null
> > @@ -1,112 +0,0 @@
> > -#!/bin/sh
> > -
> > -test_description='test oidmap'
> > -
> > -TEST_PASSES_SANITIZE_LEAK=true
> > -. ./test-lib.sh
> > -
> > -# This purposefully is very similar to t0011-hashmap.sh
> > -
> > -test_oidmap () {
> > -	echo "$1" | test-tool oidmap $3 >actual &&
> > -	echo "$2" >expect &&
> > -	test_cmp expect actual
> > -}
> > -
> > -
> > -test_expect_success 'setup' '
> > -
> > -	test_commit one &&
> > -	test_commit two &&
> > -	test_commit three &&
> > -	test_commit four
> > -
> > -'
> > -
> > -test_expect_success 'put' '
> > -
> > -test_oidmap "put one 1
> > -put two 2
> > -put invalidOid 4
> > -put three 3" "NULL
> > -NULL
> > -Unknown oid: invalidOid
> > -NULL"
> > -
> > -'
> > -
> > -test_expect_success 'replace' '
> > -
> > -test_oidmap "put one 1
> > -put two 2
> > -put three 3
> > -put invalidOid 4
> > -put two deux
> > -put one un" "NULL
> > -NULL
> > -NULL
> > -Unknown oid: invalidOid
> > -2
> > -1"
> > -
> > -'
> > -
> > -test_expect_success 'get' '
> > -
> > -test_oidmap "put one 1
> > -put two 2
> > -put three 3
> > -get two
> > -get four
> > -get invalidOid
> > -get one" "NULL
> > -NULL
> > -NULL
> > -2
> > -NULL
> > -Unknown oid: invalidOid
> > -1"
> > -
> > -'
> > -
> > -test_expect_success 'remove' '
> > -
> > -test_oidmap "put one 1
> > -put two 2
> > -put three 3
> > -remove one
> > -remove two
> > -remove invalidOid
> > -remove four" "NULL
> > -NULL
> > -NULL
> > -1
> > -2
> > -Unknown oid: invalidOid
> > -NULL"
> > -
> > -'
> > -
> > -test_expect_success 'iterate' '
> > -	test-tool oidmap >actual.raw <<-\EOF &&
> > -	put one 1
> > -	put two 2
> > -	put three 3
> > -	iterate
> > -	EOF
> > -
> > -	# sort "expect" too so we do not rely on the order of particular oids
> > -	sort >expect <<-EOF &&
> > -	NULL
> > -	NULL
> > -	NULL
> > -	$(git rev-parse one) 1
> > -	$(git rev-parse two) 2
> > -	$(git rev-parse three) 3
> > -	EOF
> > -
> > -	sort <actual.raw >actual &&
> > -	test_cmp expect actual
> > -'
> > -
> > -test_done
> > diff --git a/t/unit-tests/t-oidmap.c b/t/unit-tests/t-oidmap.c
> > new file mode 100644
> > index 0000000000..9b98a3ed09
> > --- /dev/null
> > +++ b/t/unit-tests/t-oidmap.c
> > @@ -0,0 +1,165 @@
> > +#include "test-lib.h"
> > +#include "lib-oid.h"
> > +#include "oidmap.h"
> > +#include "hash.h"
> > +#include "hex.h"
> > +
> > +/*
> > + * elements we will put in oidmap structs are made of a key: the entry.oid
> > + * field, which is of type struct object_id, and a value: the name field (could
> > + * be a refname for example)
> > + */
> > +struct test_entry {
> > +	struct oidmap_entry entry;
> > +	char name[FLEX_ARRAY];
> > +};
> > +
> > +static const char *key_val[][2] = { { "11", "one" },
> > +				    { "22", "two" },
> > +				    { "33", "three" } };
> > +
> > +static int put_and_check_null(struct oidmap *map, const char *hex,
> > +			      const char *entry_name)
> > +{
> > +	struct test_entry *entry;
> > +
> > +	FLEX_ALLOC_STR(entry, name, entry_name);
> > +	if (get_oid_arbitrary_hex(hex, &entry->entry.oid))
> > +		return -1;
> > +	if (!check(oidmap_put(map, entry) == NULL))
> > +		return -1;
> > +	return 0;
> > +}
> > +
> > +static void setup(void (*f)(struct oidmap *map))
> > +{
> > +	struct oidmap map = OIDMAP_INIT;
> > +	int ret = 0;
> > +
> > +	for (size_t i = 0; i < ARRAY_SIZE(key_val); i++)
> > +		if ((ret = put_and_check_null(&map, key_val[i][0],
> > +					      key_val[i][1])))
> > +			break;
> > +
> > +	if (!ret)
> > +		f(&map);
> > +	oidmap_free(&map, 1);
> > +}
> > +
> > +static void t_replace(struct oidmap *map)
> > +{
> > +	struct test_entry *entry, *prev;
> > +
> > +	FLEX_ALLOC_STR(entry, name, "un");
> > +	if (get_oid_arbitrary_hex("11", &entry->entry.oid))
> > +		return;
> > +	prev = oidmap_put(map, entry);
> > +	if (!check(prev != NULL))
> > +		return;
> > +	check_str(prev->name, "one");
> > +	free(prev);
> > +
> > +	FLEX_ALLOC_STR(entry, name, "deux");
> > +	if (get_oid_arbitrary_hex("22", &entry->entry.oid))
> > +		return;
> > +	prev = oidmap_put(map, entry);
> > +	if (!check(prev != NULL))
> > +		return;
> > +	check_str(prev->name, "two");
> > +	free(prev);
> > +}
> > +
> > +static void t_get(struct oidmap *map)
> > +{
> > +	struct test_entry *entry;
> > +	struct object_id oid;
> > +
> > +	if (get_oid_arbitrary_hex("22", &oid))
> > +		return;
> > +	entry = oidmap_get(map, &oid);
> > +	if (!check(entry != NULL))
> > +		return;
> > +	check_str(entry->name, "two");
> > +
> > +	if (get_oid_arbitrary_hex("44", &oid))
> > +		return;
> > +	check(oidmap_get(map, &oid) == NULL);
> > +
> > +	if (get_oid_arbitrary_hex("11", &oid))
> > +		return;
> > +	entry = oidmap_get(map, &oid);
> > +	if (!check(entry != NULL))
> > +		return;
> > +	check_str(entry->name, "one");
> > +}
> > +
> > +static void t_remove(struct oidmap *map)
> > +{
> > +	struct test_entry *entry;
> > +	struct object_id oid;
> > +
> > +	if (get_oid_arbitrary_hex("11", &oid))
> > +		return;
> > +	entry = oidmap_remove(map, &oid);
> > +	if (!check(entry != NULL))
> > +		return;
> > +	check_str(entry->name, "one");
> > +	check(oidmap_get(map, &oid) == NULL);
> > +	free(entry);
> > +
> > +	if (get_oid_arbitrary_hex("22", &oid))
> > +		return;
> > +	entry = oidmap_remove(map, &oid);
> > +	if (!check(entry != NULL))
> > +		return;
> > +	check_str(entry->name, "two");
> > +	check(oidmap_get(map, &oid) == NULL);
> > +	free(entry);
> > +
> > +	if (get_oid_arbitrary_hex("44", &oid))
> > +		return;
> > +	check(oidmap_remove(map, &oid) == NULL);
> > +}
> > +
> > +static int key_val_contains(struct test_entry *entry)
> > +{
> > +	/* the test is small enough to be able to bear O(n) */
> > +	for (size_t i = 0; i < ARRAY_SIZE(key_val); i++) {
> > +		if (!strcmp(key_val[i][1], entry->name)) {
> > +			struct object_id oid;
> > +			if (get_oid_arbitrary_hex(key_val[i][0], &oid))
> > +				return -1;
> > +			if (oideq(&entry->entry.oid, &oid))
> > +				return 0;
> > +		}
> > +	}
> > +	return 1;
> > +}
> > +
> > +static void t_iterate(struct oidmap *map)
> > +{
> > +	struct oidmap_iter iter;
> > +	struct test_entry *entry;
> > +	int ret;
> > +
> > +	oidmap_iter_init(map, &iter);
> > +	while ((entry = oidmap_iter_next(&iter))) {
> > +		if (!check_int((ret = key_val_contains(entry)), ==, 0)) {
> > +			if (ret == -1)
> > +				return;
> > +			test_msg("obtained entry was not given in the input\n"
> > +				 "  name: %s\n   oid: %s\n",
> > +				 entry->name, oid_to_hex(&entry->entry.oid));
> > +		}
> > +	}
> > +	check_int(hashmap_get_size(&map->map), ==, ARRAY_SIZE(key_val));
> > +}
> > +
> > +int cmd_main(int argc UNUSED, const char **argv UNUSED)
> > +{
> > +	TEST(setup(t_replace), "replace works");
> > +	TEST(setup(t_get), "get works");
> > +	TEST(setup(t_remove), "remove works");
> > +	TEST(setup(t_iterate), "iterate works");
> > +	return test_done();
> > +}
> > -- 
> > 2.45.2
> > 
> > 






[Index of Archives]     [Linux Kernel Development]     [Gcc Help]     [IETF Annouce]     [DCCP]     [Netdev]     [Networking]     [Security]     [V4L]     [Bugtraq]     [Yosemite]     [MIPS Linux]     [ARM Linux]     [Linux Security]     [Linux RAID]     [Linux SCSI]     [Fedora Users]

  Powered by Linux