Blame library/test.c

Packit 8586cb
/*
Packit 8586cb
 * Copyright (c) 2013, Red Hat Inc.
Packit 8586cb
 *
Packit 8586cb
 * Redistribution and use in source and binary forms, with or without
Packit 8586cb
 * modification, are permitted provided that the following conditions
Packit 8586cb
 * are met:
Packit 8586cb
 *
Packit 8586cb
 *     * Redistributions of source code must retain the above
Packit 8586cb
 *       copyright notice, this list of conditions and the
Packit 8586cb
 *       following disclaimer.
Packit 8586cb
 *     * Redistributions in binary form must reproduce the
Packit 8586cb
 *       above copyright notice, this list of conditions and
Packit 8586cb
 *       the following disclaimer in the documentation and/or
Packit 8586cb
 *       other materials provided with the distribution.
Packit 8586cb
 *     * The names of contributors to this software may not be
Packit 8586cb
 *       used to endorse or promote products derived from this
Packit 8586cb
 *       software without specific prior written permission.
Packit 8586cb
 *
Packit 8586cb
 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
Packit 8586cb
 * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
Packit 8586cb
 * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
Packit 8586cb
 * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
Packit 8586cb
 * COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
Packit 8586cb
 * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING,
Packit 8586cb
 * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS
Packit 8586cb
 * OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED
Packit 8586cb
 * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
Packit 8586cb
 * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF
Packit 8586cb
 * THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH
Packit 8586cb
 * DAMAGE.
Packit 8586cb
 *
Packit 8586cb
 * Author: Stef Walter <stefw@redhat.com>
Packit 8586cb
 */
Packit 8586cb
Packit 8586cb
#include "config.h"
Packit 8586cb
Packit 8586cb
#define TEST_SOURCE 1
Packit 8586cb
Packit 8586cb
#include "test.h"
Packit 8586cb
Packit 8586cb
#include <assert.h>
Packit 8586cb
#include <setjmp.h>
Packit 8586cb
#include <stdarg.h>
Packit 8586cb
#include <stdio.h>
Packit 8586cb
#include <stdlib.h>
Packit 8586cb
#include <string.h>
Packit 8586cb
Packit 8586cb
enum {
Packit 8586cb
	FIXTURE,
Packit 8586cb
	TEST,
Packit 8586cb
};
Packit 8586cb
Packit 8586cb
typedef void (*func_with_arg) (void *);
Packit 8586cb
Packit 8586cb
typedef struct _test_item {
Packit 8586cb
	int type;
Packit 8586cb
Packit 8586cb
	union {
Packit 8586cb
		struct {
Packit 8586cb
			char name[1024];
Packit 8586cb
			func_with_arg func;
Packit 8586cb
			void *argument;
Packit 8586cb
			int failed;
Packit 8586cb
		} test;
Packit 8586cb
		struct {
Packit 8586cb
			func_with_arg setup;
Packit 8586cb
			func_with_arg teardown;
Packit 8586cb
		} fix;
Packit 8586cb
	} x;
Packit 8586cb
Packit 8586cb
	struct _test_item *next;
Packit 8586cb
} test_item;
Packit 8586cb
Packit 8586cb
struct {
Packit 8586cb
	test_item *suite;
Packit 8586cb
	test_item *last;
Packit 8586cb
	int number;
Packit 8586cb
	jmp_buf jump;
Packit 8586cb
} gl = { NULL, NULL, 0, };
Packit 8586cb
Packit 8586cb
void
Packit 8586cb
test_fail (const char *filename,
Packit 8586cb
           int line,
Packit 8586cb
           const char *function,
Packit 8586cb
           const char *message,
Packit 8586cb
           ...)
Packit 8586cb
{
Packit 8586cb
	const char *pos;
Packit 8586cb
	char *output;
Packit 8586cb
	char *from;
Packit 8586cb
	char *next;
Packit 8586cb
	va_list va;
Packit 8586cb
Packit 8586cb
	assert (gl.last != NULL);
Packit 8586cb
	assert (gl.last->type == TEST);
Packit 8586cb
	gl.last->x.test.failed = 1;
Packit 8586cb
Packit 8586cb
	printf ("not ok %d %s\n", gl.number, gl.last->x.test.name);
Packit 8586cb
Packit 8586cb
	va_start (va, message);
Packit 8586cb
	if (vasprintf (&output, message, va) < 0)
Packit 8586cb
		assert (0 && "vasprintf() failed");
Packit 8586cb
	va_end (va);
Packit 8586cb
Packit 8586cb
	for (from = output; from != NULL; ) {
Packit 8586cb
		next = strchr (from, '\n');
Packit 8586cb
		if (next) {
Packit 8586cb
			next[0] = '\0';
Packit 8586cb
			next += 1;
Packit 8586cb
		}
Packit 8586cb
Packit 8586cb
		printf ("# %s\n", from);
Packit 8586cb
		from = next;
Packit 8586cb
	}
Packit 8586cb
Packit 8586cb
	pos = strrchr (filename, '/');
Packit 8586cb
	if (pos != NULL && pos[1] != '\0')
Packit 8586cb
		filename = pos + 1;
Packit 8586cb
Packit 8586cb
	printf ("# in %s() at %s:%d\n", function, filename, line);
Packit 8586cb
Packit 8586cb
	free (output);
Packit 8586cb
}
Packit 8586cb
Packit 8586cb
static void
Packit 8586cb
test_push (test_item *it)
Packit 8586cb
{
Packit 8586cb
	test_item *item;
Packit 8586cb
Packit 8586cb
	item = calloc (1, sizeof (test_item));
Packit 8586cb
	assert (item != NULL);
Packit 8586cb
	memcpy (item, it, sizeof (test_item));
Packit 8586cb
Packit 8586cb
	if (!gl.suite)
Packit 8586cb
		gl.suite = item;
Packit 8586cb
	if (gl.last)
Packit 8586cb
		gl.last->next = item;
Packit 8586cb
	gl.last = item;
Packit 8586cb
}
Packit 8586cb
Packit 8586cb
void
Packit 8586cb
test_func (void (* function) (void),
Packit 8586cb
           const char *name,
Packit 8586cb
           ...)
Packit 8586cb
{
Packit 8586cb
	test_item item = { TEST, };
Packit 8586cb
	va_list va;
Packit 8586cb
Packit 8586cb
	item.x.test.func = (func_with_arg)function;
Packit 8586cb
Packit 8586cb
	va_start (va, name);
Packit 8586cb
	vsnprintf (item.x.test.name, sizeof (item.x.test.name), name, va);
Packit 8586cb
	va_end (va);
Packit 8586cb
Packit 8586cb
	test_push (&item);
Packit 8586cb
}
Packit 8586cb
Packit 8586cb
void
Packit 8586cb
test_funcx (void (* function) (void *),
Packit 8586cb
            void *argument,
Packit 8586cb
            const char *name,
Packit 8586cb
            ...)
Packit 8586cb
{
Packit 8586cb
	test_item item = { TEST, };
Packit 8586cb
	va_list va;
Packit 8586cb
Packit 8586cb
	item.type = TEST;
Packit 8586cb
	item.x.test.func = function;
Packit 8586cb
	item.x.test.argument = argument;
Packit 8586cb
Packit 8586cb
	va_start (va, name);
Packit 8586cb
	vsnprintf (item.x.test.name, sizeof (item.x.test.name), name, va);
Packit 8586cb
	va_end (va);
Packit 8586cb
Packit 8586cb
	test_push (&item);
Packit 8586cb
}
Packit 8586cb
Packit 8586cb
void
Packit 8586cb
test_fixture (void (* setup) (void *),
Packit 8586cb
              void (* teardown) (void *))
Packit 8586cb
{
Packit 8586cb
	test_item item;
Packit 8586cb
Packit 8586cb
	item.type = FIXTURE;
Packit 8586cb
	item.x.fix.setup = setup;
Packit 8586cb
	item.x.fix.teardown = teardown;
Packit 8586cb
Packit 8586cb
	test_push (&item);
Packit 8586cb
}
Packit 8586cb
Packit 8586cb
int
Packit 8586cb
test_run (int argc,
Packit 8586cb
          char **argv)
Packit 8586cb
{
Packit 8586cb
	test_item *fixture = NULL;
Packit 8586cb
	test_item *item;
Packit 8586cb
	test_item *next;
Packit 8586cb
	int count;
Packit 8586cb
	int ret = 0;
Packit 8586cb
Packit 8586cb
	assert (gl.number == 0);
Packit 8586cb
	gl.last = NULL;
Packit 8586cb
Packit 8586cb
	for (item = gl.suite, count = 0; item != NULL; item = item->next) {
Packit 8586cb
		if (item->type == TEST)
Packit 8586cb
			count++;
Packit 8586cb
	}
Packit 8586cb
Packit 8586cb
	if (count == 0) {
Packit 8586cb
		printf ("1..0 # No tests\n");
Packit 8586cb
		return 0;
Packit 8586cb
	}
Packit 8586cb
Packit 8586cb
	printf ("1..%d\n", count);
Packit 8586cb
Packit 8586cb
	for (item = gl.suite, gl.number = 0; item != NULL; item = item->next) {
Packit 8586cb
		if (item->type == FIXTURE) {
Packit 8586cb
			fixture = item;
Packit 8586cb
			continue;
Packit 8586cb
		}
Packit 8586cb
Packit 8586cb
		assert (item->type == TEST);
Packit 8586cb
		gl.last = item;
Packit 8586cb
		gl.number++;
Packit 8586cb
Packit 8586cb
		if (setjmp (gl.jump) == 0) {
Packit 8586cb
			if (fixture && fixture->x.fix.setup)
Packit 8586cb
				(fixture->x.fix.setup) (item->x.test.argument);
Packit 8586cb
Packit 8586cb
			assert (item->x.test.func);
Packit 8586cb
			(item->x.test.func)(item->x.test.argument);
Packit 8586cb
Packit 8586cb
			if (fixture && fixture->x.fix.teardown)
Packit 8586cb
				(fixture->x.fix.teardown) (item->x.test.argument);
Packit 8586cb
Packit 8586cb
			printf ("ok %d %s\n", gl.number, item->x.test.name);
Packit 8586cb
		}
Packit 8586cb
Packit 8586cb
		gl.last = NULL;
Packit 8586cb
	}
Packit 8586cb
Packit 8586cb
	for (item = gl.suite; item != NULL; item = next) {
Packit 8586cb
		if (item->type == TEST) {
Packit 8586cb
			if (item->x.test.failed)
Packit 8586cb
				ret++;
Packit 8586cb
		}
Packit 8586cb
Packit 8586cb
		next = item->next;
Packit 8586cb
		free (item);
Packit 8586cb
	}
Packit 8586cb
Packit 8586cb
	gl.suite = NULL;
Packit 8586cb
	gl.last = 0;
Packit 8586cb
	gl.number = 0;
Packit 8586cb
	return ret;
Packit 8586cb
}