gtester: implemented XML logging.

* glib/gtester.c: log test messages to XML output file. beautified normal test
result output.

* glib/gtestframework.c: fixed GTimer leak.

* glib/tests/Makefile.am: start gtester with --verbose.

svn path=/trunk/; revision=5903
This commit is contained in:
Tim Janik 2007-11-20 15:00:51 +00:00
parent 6bc38c46e7
commit f347900eaf
3 changed files with 142 additions and 29 deletions

View File

@ -18,6 +18,7 @@
* Boston, MA 02111-1307, USA. * Boston, MA 02111-1307, USA.
*/ */
#include <glib.h> #include <glib.h>
#include <gstdio.h>
#include <string.h> #include <string.h>
#include <stdlib.h> #include <stdlib.h>
#include <unistd.h> #include <unistd.h>
@ -34,19 +35,102 @@ static void parse_args (gint *argc_p,
/* --- variables --- */ /* --- variables --- */
static GIOChannel *ioc_report = NULL; static GIOChannel *ioc_report = NULL;
static gboolean subtest_running = FALSE; static gboolean gtester_quiet = FALSE;
static gboolean subtest_io_pending = FALSE;
static gboolean gtester_quiet = TRUE;
static gboolean gtester_verbose = FALSE; static gboolean gtester_verbose = FALSE;
static gboolean gtester_list_tests = FALSE; static gboolean gtester_list_tests = FALSE;
static gboolean subtest_running = FALSE;
static gboolean subtest_io_pending = FALSE;
static gboolean subtest_faill = TRUE;
static gboolean subtest_quiet = TRUE;
static gboolean subtest_verbose = FALSE;
static gboolean subtest_mode_fatal = FALSE; static gboolean subtest_mode_fatal = FALSE;
static gboolean subtest_mode_perf = FALSE; static gboolean subtest_mode_perf = FALSE;
static gboolean subtest_mode_quick = TRUE; static gboolean subtest_mode_quick = TRUE;
static const gchar *subtest_seedstr = NULL; static const gchar *subtest_seedstr = NULL;
static GSList *subtest_paths = NULL; static GSList *subtest_paths = NULL;
static const gchar *outpu_filename = NULL; static gboolean testcase_open = FALSE;
static const gchar *output_filename = NULL;
static guint log_indent = 0;
static gint log_fd = -1;
/* --- functions --- */ /* --- functions --- */
static const char*
sindent (guint n)
{
static const char spaces[] = " ";
int l = sizeof (spaces);
n = MIN (n, l - 1);
return spaces + l - n;
}
static void G_GNUC_PRINTF (1, 2)
test_log_printfe (const char *format,
...)
{
char *result;
int r;
va_list args;
va_start (args, format);
result = g_markup_vprintf_escaped (format, args);
va_end (args);
do
r = write (log_fd, result, strlen (result));
while (r < 0 && errno == EINTR);
g_free (result);
}
static void
test_log_msg (GTestLogMsg *msg)
{
switch (msg->log_type)
{
case G_TEST_LOG_NONE:
break;
case G_TEST_LOG_ERROR:
g_printerr ("%s\n", msg->strings[0]);
break;
case G_TEST_LOG_START_BINARY:
test_log_printfe ("%s<binary file=\"%s\"/>\n", sindent (log_indent), msg->strings[0]);
test_log_printfe ("%s<random-seed>%s</random-seed>\n", sindent (log_indent), msg->strings[1]);
break;
case G_TEST_LOG_LIST_CASE:
g_print ("%s\n", msg->strings[0]);
break;
case G_TEST_LOG_START_CASE:
if (gtester_verbose)
{
gchar *sc = g_strconcat (msg->strings[0], ":", NULL);
gchar *sleft = g_strdup_printf ("%-68s", sc);
g_free (sc);
g_print ("%70s ", sleft);
g_free (sleft);
}
g_return_if_fail (testcase_open == 0);
testcase_open++;
test_log_printfe ("%s<testcase path=\"%s\">\n", sindent (log_indent), msg->strings[0]);
log_indent += 2;
break;
case G_TEST_LOG_STOP_CASE:
g_return_if_fail (testcase_open > 0);
test_log_printfe ("%s<duration>%.6Lf</duration>\n", sindent (log_indent), msg->nums[2]);
test_log_printfe ("%s<status exit-status=\"%d\" n-forks=\"%d\"/>\n",
sindent (log_indent), (int) msg->nums[0], (int) msg->nums[1]);
log_indent -= 2;
test_log_printfe ("%s</testcase>\n", sindent (log_indent));
testcase_open--;
if (gtester_verbose)
g_print ("OK\n");
break;
case G_TEST_LOG_MIN_RESULT:
case G_TEST_LOG_MAX_RESULT:
test_log_printfe ("%s<performance minimize=\"%d\" maximize=\"%d\" value=\"%.16Lg\">\n",
sindent (log_indent), msg->log_type == G_TEST_LOG_MIN_RESULT, msg->log_type == G_TEST_LOG_MAX_RESULT, msg->nums[0]);
test_log_printfe ("%s%s\n", sindent (log_indent + 2), msg->strings[0]);
test_log_printfe ("%s</performance>\n", sindent (log_indent));
break;
}
}
static gboolean static gboolean
child_report_cb (GIOChannel *source, child_report_cb (GIOChannel *source,
GIOCondition condition, GIOCondition condition,
@ -69,19 +153,7 @@ child_report_cb (GIOChannel *source,
msg = g_test_log_buffer_pop (tlb); msg = g_test_log_buffer_pop (tlb);
if (msg) if (msg)
{ {
guint ui; test_log_msg (msg);
/* print message, this should be written to an XML log file */
g_printerr ("{*GTLOG(%s)", g_test_log_type_name (msg->log_type));
for (ui = 0; ui < msg->n_strings; ui++)
g_printerr (":{%s}", msg->strings[ui]);
if (msg->n_nums)
{
g_printerr (":(");
for (ui = 0; ui < msg->n_nums; ui++)
g_printerr ("%s%.16Lg", ui ? ";" : "", msg->nums[ui]);
g_printerr (")");
}
g_printerr (":GTLOG*}\n");
g_test_log_msg_free (msg); g_test_log_msg_free (msg);
} }
} }
@ -129,8 +201,8 @@ unset_cloexec_fdp (gpointer fdp_data)
while (r < 0 && errno == EINTR); while (r < 0 && errno == EINTR);
} }
static void static gboolean
launch_test (const char *binary) launch_test_binary (const char *binary)
{ {
GTestLogBuffer *tlb; GTestLogBuffer *tlb;
GSList *slist, *free_list = NULL; GSList *slist, *free_list = NULL;
@ -146,14 +218,14 @@ launch_test (const char *binary)
g_error ("Failed to open pipe for test binary: %s: %s", binary, g_strerror (errno)); g_error ("Failed to open pipe for test binary: %s: %s", binary, g_strerror (errno));
else else
g_warning ("Failed to open pipe for test binary: %s: %s", binary, g_strerror (errno)); g_warning ("Failed to open pipe for test binary: %s: %s", binary, g_strerror (errno));
return; return FALSE;
} }
/* setup argv */ /* setup argv */
argv[i++] = binary; argv[i++] = binary;
if (gtester_quiet) if (subtest_quiet)
argv[i++] = "--quiet"; argv[i++] = "--quiet";
if (gtester_verbose) if (subtest_verbose)
argv[i++] = "--verbose"; argv[i++] = "--verbose";
// argv[i++] = "--debug-log"; // argv[i++] = "--debug-log";
argv[i++] = queue_gfree (&free_list, g_strdup_printf ("--GTestLogFD=%u", report_pipe[1])); argv[i++] = queue_gfree (&free_list, g_strdup_printf ("--GTestLogFD=%u", report_pipe[1]));
@ -188,6 +260,9 @@ launch_test (const char *binary)
free_list = NULL; free_list = NULL;
close (report_pipe[1]); close (report_pipe[1]);
if (!gtester_quiet)
g_print ("(pid=%lu)\n", (unsigned long) pid);
if (error) if (error)
{ {
close (report_pipe[0]); close (report_pipe[0]);
@ -196,7 +271,7 @@ launch_test (const char *binary)
else else
g_warning ("Failed to execute test binary: %s: %s", argv[0], error->message); g_warning ("Failed to execute test binary: %s: %s", argv[0], error->message);
g_clear_error (&error); g_clear_error (&error);
return; return FALSE;
} }
subtest_running = TRUE; subtest_running = TRUE;
@ -220,6 +295,29 @@ launch_test (const char *binary)
close (report_pipe[0]); close (report_pipe[0]);
g_test_log_buffer_free (tlb); g_test_log_buffer_free (tlb);
return TRUE;
}
static void
launch_test (const char *binary)
{
gboolean success;
GTimer *btimer = g_timer_new();
subtest_faill = FALSE;
if (!gtester_quiet)
g_print ("TEST: %s... ", binary);
test_log_printfe ("%s<testbinary path=\"%s\">\n", sindent (log_indent), binary);
log_indent += 2;
g_timer_start (btimer);
success = launch_test_binary (binary);
g_timer_stop (btimer);
test_log_printfe ("%s<duration>%.6f</duration>\n", sindent (log_indent), g_timer_elapsed (btimer, NULL));
log_indent -= 2;
test_log_printfe ("%s</testbinary>\n", sindent (log_indent));
if (!gtester_quiet)
g_print ("%s: %s\n", subtest_faill || !success ? "FAIL" : "PASS", binary);
g_timer_destroy (btimer);
} }
static void static void
@ -242,8 +340,8 @@ usage (gboolean just_version)
g_print (" -p=TESTPATH only start test cases matching TESTPATH\n"); g_print (" -p=TESTPATH only start test cases matching TESTPATH\n");
g_print (" --seed=SEEDSTRING start all tests with random number seed SEEDSTRING\n"); g_print (" --seed=SEEDSTRING start all tests with random number seed SEEDSTRING\n");
g_print (" -o=LOGFILE write the test log to LOGFILE\n"); g_print (" -o=LOGFILE write the test log to LOGFILE\n");
g_print (" -q, --quiet suppress unnecessary output\n"); g_print (" -q, --quiet suppress per test binary output\n");
g_print (" --verbose produce additional output\n"); g_print (" --verbose report success per testcase\n");
} }
static void static void
@ -297,11 +395,11 @@ parse_args (gint *argc_p,
{ {
gchar *equal = argv[i] + 2; gchar *equal = argv[i] + 2;
if (*equal == '=') if (*equal == '=')
outpu_filename = equal + 1; output_filename = equal + 1;
else if (i + 1 < argc) else if (i + 1 < argc)
{ {
argv[i++] = NULL; argv[i++] = NULL;
outpu_filename = argv[i]; output_filename = argv[i];
} }
argv[i] = NULL; argv[i] = NULL;
} }
@ -400,12 +498,26 @@ main (int argc,
return 1; return 1;
} }
if (output_filename)
{
log_fd = g_open (output_filename, O_WRONLY | O_CREAT | O_TRUNC, 0666);
if (log_fd < 0)
g_error ("Failed to open log file '%s': %s", output_filename, g_strerror (errno));
}
test_log_printfe ("<?xml version=\"1.0\"?>\n");
test_log_printfe ("%s<gtester>\n", sindent (log_indent));
log_indent += 2;
for (ui = 1; ui < argc; ui++) for (ui = 1; ui < argc; ui++)
{ {
const char *binary = argv[ui]; const char *binary = argv[ui];
launch_test (binary); launch_test (binary);
/* we only get here on success or if !subtest_mode_fatal */
} }
log_indent -= 2;
test_log_printfe ("%s</gtester>\n", sindent (log_indent));
close (log_fd);
/* we only get here on success or if !subtest_mode_fatal */
return 0; return 0;
} }

View File

@ -625,6 +625,7 @@ test_case_run (GTestCase *tc)
largs[1] = test_run_forks; largs[1] = test_run_forks;
largs[2] = g_timer_elapsed (test_run_timer, NULL); largs[2] = g_timer_elapsed (test_run_timer, NULL);
g_test_log (G_TEST_LOG_STOP_CASE, NULL, NULL, G_N_ELEMENTS (largs), largs); g_test_log (G_TEST_LOG_STOP_CASE, NULL, NULL, G_N_ELEMENTS (largs), largs);
g_timer_destroy (test_run_timer);
} }
g_free (test_run_name); g_free (test_run_name);
test_run_name = old_name; test_run_name = old_name;

View File

@ -13,6 +13,6 @@ testing_LDADD = $(progs_ldadd)
test: test:
${GTESTER} ${TEST_PROGS} ${GTESTER} --verbose ${TEST_PROGS}
.PHONY: test .PHONY: test
check-local: test check-local: test