181 lines
5.5 KiB
C
181 lines
5.5 KiB
C
|
// Copyright 2012 Google Inc.
|
||
|
// All rights reserved.
|
||
|
//
|
||
|
// Redistribution and use in source and binary forms, with or without
|
||
|
// modification, are permitted provided that the following conditions are
|
||
|
// met:
|
||
|
//
|
||
|
// * Redistributions of source code must retain the above copyright
|
||
|
// notice, this list of conditions and the following disclaimer.
|
||
|
// * Redistributions in binary form must reproduce the above copyright
|
||
|
// notice, this list of conditions and the following disclaimer in the
|
||
|
// documentation and/or other materials provided with the distribution.
|
||
|
// * Neither the name of Google Inc. nor the names of its contributors
|
||
|
// may be used to endorse or promote products derived from this software
|
||
|
// without specific prior written permission.
|
||
|
//
|
||
|
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
||
|
// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
||
|
// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
|
||
|
// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
|
||
|
// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
|
||
|
// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
||
|
// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
||
|
// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
|
||
|
// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
||
|
// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
||
|
// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||
|
|
||
|
/// \file common_inttest.h
|
||
|
/// Common integration tests for the tester binaries.
|
||
|
|
||
|
#if defined(KYUA_COMMON_INTTEST_H)
|
||
|
# error "common_inttest.h can only be defined once"
|
||
|
#endif
|
||
|
/// Include guard.
|
||
|
#define KYUA_COMMON_INTTEST_H
|
||
|
|
||
|
#if !defined(INTERFACE)
|
||
|
# error "Must define INTERFACE to the name of the tester interface"
|
||
|
#endif
|
||
|
|
||
|
#include <err.h>
|
||
|
#include <stdarg.h>
|
||
|
#include <stdbool.h>
|
||
|
#include <stdio.h>
|
||
|
#include <stdlib.h>
|
||
|
|
||
|
#include <atf-c.h>
|
||
|
|
||
|
#include "cli.h" // For the EXIT_* constants only.
|
||
|
#include "defs.h"
|
||
|
|
||
|
|
||
|
/// Path to the installed testers.
|
||
|
static const char* default_testersdir = TESTERSDIR;
|
||
|
|
||
|
|
||
|
/// Returns the name of the current tester.
|
||
|
#define TESTER_BIN "kyua-" INTERFACE "-tester"
|
||
|
|
||
|
|
||
|
/// Returns the path to the helpers.
|
||
|
///
|
||
|
/// \param tc Pointer to the caller test case, to obtain the srcdir property.
|
||
|
///
|
||
|
/// \return A dynamically-allocated string; must be released with free(3).
|
||
|
static char*
|
||
|
helpers_path(const atf_tc_t* tc)
|
||
|
{
|
||
|
const char* srcdir = atf_tc_get_config_var(tc, "srcdir");
|
||
|
const char* name = INTERFACE "_helpers";
|
||
|
|
||
|
const size_t length = strlen(srcdir) + 1 + strlen(name) + 1;
|
||
|
char* buffer = (char*)malloc(length);
|
||
|
(void)snprintf(buffer, length, "%s/%s", srcdir, name);
|
||
|
return buffer;
|
||
|
}
|
||
|
|
||
|
|
||
|
/// Returns the path to the tester.
|
||
|
///
|
||
|
/// \return A dynamically-allocated string; must be released with free(3).
|
||
|
static char*
|
||
|
tester_path(void)
|
||
|
{
|
||
|
const char* testersdir = getenv("TESTERSDIR");
|
||
|
if (testersdir == NULL)
|
||
|
testersdir = default_testersdir;
|
||
|
const char* name = TESTER_BIN;
|
||
|
|
||
|
const size_t length = strlen(testersdir) + 1 + strlen(name) + 1;
|
||
|
char* buffer = (char*)malloc(length);
|
||
|
ATF_REQUIRE(buffer != NULL);
|
||
|
(void)snprintf(buffer, length, "%s/%s", testersdir, name);
|
||
|
return buffer;
|
||
|
}
|
||
|
|
||
|
|
||
|
/// Initializes the test case metadata and the helpers.
|
||
|
///
|
||
|
/// \param [in,out] tc The test case in which to set the property.
|
||
|
/// \param uses_helpers Whether the test uses the helpers or not.
|
||
|
static void
|
||
|
setup(atf_tc_t* tc, const bool uses_helpers)
|
||
|
{
|
||
|
char* tester = tester_path();
|
||
|
if (uses_helpers) {
|
||
|
char* helpers = helpers_path(tc);
|
||
|
atf_tc_set_md_var(tc, "require.progs", "%s %s", tester, helpers);
|
||
|
free(helpers);
|
||
|
} else {
|
||
|
atf_tc_set_md_var(tc, "require.progs", "%s", tester);
|
||
|
}
|
||
|
free(tester);
|
||
|
}
|
||
|
|
||
|
|
||
|
static void execute(va_list ap) KYUA_DEFS_NORETURN;
|
||
|
|
||
|
|
||
|
/// Executes the tester with the given set of variable arguments.
|
||
|
///
|
||
|
/// \param ap List of arguments to the tester.
|
||
|
static void
|
||
|
execute(va_list ap)
|
||
|
{
|
||
|
const char* args[16];
|
||
|
|
||
|
const char** current_arg = &args[0];
|
||
|
*current_arg = TESTER_BIN;
|
||
|
++current_arg;
|
||
|
while ((*current_arg = va_arg(ap, const char*)) != NULL)
|
||
|
++current_arg;
|
||
|
|
||
|
char* program = tester_path();
|
||
|
(void)execv(program, KYUA_DEFS_UNCONST(args));
|
||
|
free(program);
|
||
|
err(111, "Failed to execute %s", program);
|
||
|
}
|
||
|
|
||
|
|
||
|
/// Executes the tester and validates its output.
|
||
|
///
|
||
|
/// \param expected_exit_status Expected exit status of the subprocess.
|
||
|
/// \param expected_stdout Expected contents of stdout.
|
||
|
/// \param expected_stderr Expected contents of stderr.
|
||
|
/// \param ... Arguments to the tester, not including the program name.
|
||
|
static void
|
||
|
check(const int expected_exit_status, const char* expected_stdout,
|
||
|
const char* expected_stderr, ...)
|
||
|
{
|
||
|
const pid_t pid = atf_utils_fork();
|
||
|
if (pid == 0) {
|
||
|
va_list ap;
|
||
|
va_start(ap, expected_stderr);
|
||
|
execute(ap);
|
||
|
va_end(ap);
|
||
|
} else {
|
||
|
atf_utils_wait(pid, expected_exit_status, expected_stdout,
|
||
|
expected_stderr);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
ATF_TC(top__missing_command);
|
||
|
ATF_TC_HEAD(top__missing_command, tc) { setup(tc, false); }
|
||
|
ATF_TC_BODY(top__missing_command, tc)
|
||
|
{
|
||
|
check(EXIT_USAGE_ERROR, "", TESTER_BIN": Must provide a command\n",
|
||
|
NULL);
|
||
|
}
|
||
|
|
||
|
|
||
|
ATF_TC(top__unknown_command);
|
||
|
ATF_TC_HEAD(top__unknown_command, tc) { setup(tc, false); }
|
||
|
ATF_TC_BODY(top__unknown_command, tc)
|
||
|
{
|
||
|
check(EXIT_USAGE_ERROR, "", TESTER_BIN": Unknown command 'foo'\n",
|
||
|
"foo", NULL);
|
||
|
}
|