202 lines
		
	
	
		
			5.0 KiB
		
	
	
	
		
			C
		
	
	
			
		
		
	
	
			202 lines
		
	
	
		
			5.0 KiB
		
	
	
	
		
			C
		
	
	
| #include "unit-test.h"
 | |
| #include "lib-reftable.h"
 | |
| #include "reftable/blocksource.h"
 | |
| #include "reftable/constants.h"
 | |
| #include "reftable/iter.h"
 | |
| #include "reftable/table.h"
 | |
| #include "strbuf.h"
 | |
| 
 | |
| void test_reftable_table__seek_once(void)
 | |
| {
 | |
| 	struct reftable_ref_record records[] = {
 | |
| 		{
 | |
| 			.refname = (char *) "refs/heads/main",
 | |
| 			.value_type = REFTABLE_REF_VAL1,
 | |
| 			.value.val1 = { 42 },
 | |
| 		},
 | |
| 	};
 | |
| 	struct reftable_block_source source = { 0 };
 | |
| 	struct reftable_ref_record ref = { 0 };
 | |
| 	struct reftable_iterator it = { 0 };
 | |
| 	struct reftable_table *table;
 | |
| 	struct reftable_buf buf = REFTABLE_BUF_INIT;
 | |
| 	int ret;
 | |
| 
 | |
| 	cl_reftable_write_to_buf(&buf, records, ARRAY_SIZE(records), NULL, 0, NULL);
 | |
| 	block_source_from_buf(&source, &buf);
 | |
| 
 | |
| 	ret = reftable_table_new(&table, &source, "name");
 | |
| 	cl_assert(!ret);
 | |
| 
 | |
| 	reftable_table_init_ref_iterator(table, &it);
 | |
| 	ret = reftable_iterator_seek_ref(&it, "");
 | |
| 	cl_assert(!ret);
 | |
| 	ret = reftable_iterator_next_ref(&it, &ref);
 | |
| 	cl_assert(!ret);
 | |
| 
 | |
| 	ret = reftable_ref_record_equal(&ref, &records[0],
 | |
| 					REFTABLE_HASH_SIZE_SHA1);
 | |
| 	cl_assert_equal_i(ret, 1);
 | |
| 
 | |
| 	ret = reftable_iterator_next_ref(&it, &ref);
 | |
| 	cl_assert_equal_i(ret, 1);
 | |
| 
 | |
| 	reftable_ref_record_release(&ref);
 | |
| 	reftable_iterator_destroy(&it);
 | |
| 	reftable_table_decref(table);
 | |
| 	reftable_buf_release(&buf);
 | |
| }
 | |
| 
 | |
| void test_reftable_table__reseek(void)
 | |
| {
 | |
| 	struct reftable_ref_record records[] = {
 | |
| 		{
 | |
| 			.refname = (char *) "refs/heads/main",
 | |
| 			.value_type = REFTABLE_REF_VAL1,
 | |
| 			.value.val1 = { 42 },
 | |
| 		},
 | |
| 	};
 | |
| 	struct reftable_block_source source = { 0 };
 | |
| 	struct reftable_ref_record ref = { 0 };
 | |
| 	struct reftable_iterator it = { 0 };
 | |
| 	struct reftable_table *table;
 | |
| 	struct reftable_buf buf = REFTABLE_BUF_INIT;
 | |
| 	int ret;
 | |
| 
 | |
| 	cl_reftable_write_to_buf(&buf, records, ARRAY_SIZE(records),
 | |
| 				 NULL, 0, NULL);
 | |
| 	block_source_from_buf(&source, &buf);
 | |
| 
 | |
| 	ret = reftable_table_new(&table, &source, "name");
 | |
| 	cl_assert(!ret);
 | |
| 
 | |
| 	reftable_table_init_ref_iterator(table, &it);
 | |
| 
 | |
| 	for (size_t i = 0; i < 5; i++) {
 | |
| 		ret = reftable_iterator_seek_ref(&it, "");
 | |
| 		cl_assert(!ret);
 | |
| 		ret = reftable_iterator_next_ref(&it, &ref);
 | |
| 		cl_assert(!ret);
 | |
| 
 | |
| 		ret = reftable_ref_record_equal(&ref, &records[0], REFTABLE_HASH_SIZE_SHA1);
 | |
| 		cl_assert_equal_i(ret, 1);
 | |
| 
 | |
| 		ret = reftable_iterator_next_ref(&it, &ref);
 | |
| 		cl_assert_equal_i(ret, 1);
 | |
| 	}
 | |
| 
 | |
| 	reftable_ref_record_release(&ref);
 | |
| 	reftable_iterator_destroy(&it);
 | |
| 	reftable_table_decref(table);
 | |
| 	reftable_buf_release(&buf);
 | |
| }
 | |
| 
 | |
| void test_reftable_table__block_iterator(void)
 | |
| {
 | |
| 	struct reftable_block_source source = { 0 };
 | |
| 	struct reftable_table_iterator it = { 0 };
 | |
| 	struct reftable_ref_record *records;
 | |
| 	const struct reftable_block *block;
 | |
| 	struct reftable_table *table;
 | |
| 	struct reftable_buf buf = REFTABLE_BUF_INIT;
 | |
| 	struct {
 | |
| 		uint8_t block_type;
 | |
| 		uint16_t header_off;
 | |
| 		uint16_t restart_count;
 | |
| 		uint16_t record_count;
 | |
| 	} expected_blocks[] = {
 | |
| 		{
 | |
| 			.block_type = REFTABLE_BLOCK_TYPE_REF,
 | |
| 			.header_off = 24,
 | |
| 			.restart_count = 10,
 | |
| 			.record_count = 158,
 | |
| 		},
 | |
| 		{
 | |
| 			.block_type = REFTABLE_BLOCK_TYPE_REF,
 | |
| 			.restart_count = 10,
 | |
| 			.record_count = 159,
 | |
| 		},
 | |
| 		{
 | |
| 			.block_type = REFTABLE_BLOCK_TYPE_REF,
 | |
| 			.restart_count = 10,
 | |
| 			.record_count = 159,
 | |
| 		},
 | |
| 		{
 | |
| 			.block_type = REFTABLE_BLOCK_TYPE_REF,
 | |
| 			.restart_count = 2,
 | |
| 			.record_count = 24,
 | |
| 		},
 | |
| 		{
 | |
| 			.block_type = REFTABLE_BLOCK_TYPE_INDEX,
 | |
| 			.restart_count = 1,
 | |
| 			.record_count = 4,
 | |
| 		},
 | |
| 		{
 | |
| 			.block_type = REFTABLE_BLOCK_TYPE_OBJ,
 | |
| 			.restart_count = 1,
 | |
| 			.record_count = 1,
 | |
| 		},
 | |
| 	};
 | |
| 	const size_t nrecords = 500;
 | |
| 	int ret;
 | |
| 
 | |
| 	REFTABLE_CALLOC_ARRAY(records, nrecords);
 | |
| 	for (size_t i = 0; i < nrecords; i++) {
 | |
| 		records[i].value_type = REFTABLE_REF_VAL1;
 | |
| 		records[i].refname = xstrfmt("refs/heads/branch-%03"PRIuMAX,
 | |
| 					     (uintmax_t) i);
 | |
| 	}
 | |
| 
 | |
| 	cl_reftable_write_to_buf(&buf, records, nrecords, NULL, 0, NULL);
 | |
| 	block_source_from_buf(&source, &buf);
 | |
| 
 | |
| 	ret = reftable_table_new(&table, &source, "name");
 | |
| 	cl_assert(!ret);
 | |
| 
 | |
| 	ret = reftable_table_iterator_init(&it, table);
 | |
| 	cl_assert(!ret);
 | |
| 
 | |
| 	for (size_t i = 0; i < ARRAY_SIZE(expected_blocks); i++) {
 | |
| 		struct reftable_iterator record_it = { 0 };
 | |
| 		struct reftable_record record = {
 | |
| 			.type = expected_blocks[i].block_type,
 | |
| 		};
 | |
| 
 | |
| 		ret = reftable_table_iterator_next(&it, &block);
 | |
| 		cl_assert(!ret);
 | |
| 
 | |
| 		cl_assert_equal_i(block->block_type,
 | |
| 				  expected_blocks[i].block_type);
 | |
| 		cl_assert_equal_i(block->header_off,
 | |
| 				  expected_blocks[i].header_off);
 | |
| 		cl_assert_equal_i(block->restart_count,
 | |
| 				  expected_blocks[i].restart_count);
 | |
| 
 | |
| 		ret = reftable_block_init_iterator(block, &record_it);
 | |
| 		cl_assert(!ret);
 | |
| 
 | |
| 		for (size_t j = 0; ; j++) {
 | |
| 			ret = iterator_next(&record_it, &record);
 | |
| 			if (ret > 0) {
 | |
| 				cl_assert_equal_i(j,
 | |
| 						  expected_blocks[i].record_count);
 | |
| 				break;
 | |
| 			}
 | |
| 			cl_assert(!ret);
 | |
| 		}
 | |
| 
 | |
| 		reftable_iterator_destroy(&record_it);
 | |
| 		reftable_record_release(&record);
 | |
| 	}
 | |
| 
 | |
| 	ret = reftable_table_iterator_next(&it, &block);
 | |
| 	cl_assert_equal_i(ret, 1);
 | |
| 
 | |
| 	for (size_t i = 0; i < nrecords; i++)
 | |
| 		reftable_free(records[i].refname);
 | |
| 	reftable_table_iterator_release(&it);
 | |
| 	reftable_table_decref(table);
 | |
| 	reftable_buf_release(&buf);
 | |
| 	reftable_free(records);
 | |
| }
 |