520 lines
13 KiB
C++
520 lines
13 KiB
C++
/*
|
|
|
|
Copyright (c) 2008, Arvid Norberg
|
|
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 the author 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.
|
|
|
|
*/
|
|
|
|
#include <iostream>
|
|
#include <boost/config.hpp>
|
|
#include <fcntl.h>
|
|
#include <stdio.h>
|
|
#include <stdlib.h> // for exit()
|
|
#include "libtorrent/address.hpp"
|
|
#include "libtorrent/socket.hpp"
|
|
#include "setup_transfer.hpp" // for _g_test_failures
|
|
#include "test.hpp"
|
|
#include "dht_server.hpp" // for stop_dht
|
|
#include "peer_server.hpp" // for stop_peer
|
|
#include "udp_tracker.hpp" // for stop_udp_tracker
|
|
|
|
#include "libtorrent/assert.hpp"
|
|
#include "libtorrent/file.hpp"
|
|
#include "libtorrent/aux_/escape_string.hpp"
|
|
#include <signal.h>
|
|
|
|
#ifdef WIN32
|
|
#include <windows.h> // fot SetErrorMode
|
|
#include <io.h> // for _dup and _dup2
|
|
#include <process.h> // for _getpid
|
|
#include <crtdbg.h>
|
|
|
|
#define dup _dup
|
|
#define dup2 _dup2
|
|
|
|
#else
|
|
|
|
#include <unistd.h> // for getpid()
|
|
|
|
#endif
|
|
|
|
using namespace libtorrent;
|
|
|
|
// these are global so we can restore them on abnormal exits and print stuff
|
|
// out, such as the log
|
|
int old_stdout = -1;
|
|
int old_stderr = -1;
|
|
bool redirect_stdout = true;
|
|
bool redirect_stderr = false; // TODO: TEMPORARY!
|
|
bool keep_files = false;
|
|
|
|
extern int _g_test_idx;
|
|
|
|
// the current tests file descriptor
|
|
unit_test_t* current_test = NULL;
|
|
|
|
void output_test_log_to_terminal()
|
|
{
|
|
if (current_test == NULL || old_stdout == -1 || old_stderr == -1
|
|
|| !redirect_stdout || current_test->output == NULL)
|
|
return;
|
|
|
|
fflush(stdout);
|
|
fflush(stderr);
|
|
dup2(old_stdout, fileno(stdout));
|
|
dup2(old_stderr, fileno(stderr));
|
|
|
|
fseek(current_test->output, 0, SEEK_SET);
|
|
fprintf(stdout, "\x1b[1m[%s]\x1b[0m\n\n", current_test->name);
|
|
char buf[4096];
|
|
int size = 0;
|
|
do {
|
|
size = fread(buf, 1, sizeof(buf), current_test->output);
|
|
if (size > 0) fwrite(buf, 1, size, stdout);
|
|
} while (size > 0);
|
|
}
|
|
|
|
#ifdef _WIN32
|
|
LONG WINAPI seh_exception_handler(LPEXCEPTION_POINTERS p)
|
|
{
|
|
char stack_text[10000];
|
|
|
|
#if TORRENT_USE_ASSERTS \
|
|
|| defined TORRENT_ASIO_DEBUGGING \
|
|
|| defined TORRENT_PROFILE_CALLS \
|
|
|| defined TORRENT_DEBUG_BUFFERS
|
|
print_backtrace(stack_text, sizeof(stack_text), 30
|
|
, p->ContextRecord);
|
|
#elif defined __FUNCTION__
|
|
strcat(stack_text, __FUNCTION__);
|
|
#else
|
|
stack_text[0] = 0;
|
|
strcat(stack_text, "<stack traces disabled>");
|
|
#endif
|
|
|
|
int const code = p->ExceptionRecord->ExceptionCode;
|
|
char const* name = "<unknown exception>";
|
|
switch (code)
|
|
{
|
|
#define EXC(x) case x: name = #x; break
|
|
EXC(EXCEPTION_ACCESS_VIOLATION);
|
|
EXC(EXCEPTION_ARRAY_BOUNDS_EXCEEDED);
|
|
EXC(EXCEPTION_BREAKPOINT);
|
|
EXC(EXCEPTION_DATATYPE_MISALIGNMENT);
|
|
EXC(EXCEPTION_FLT_DENORMAL_OPERAND);
|
|
EXC(EXCEPTION_FLT_DIVIDE_BY_ZERO);
|
|
EXC(EXCEPTION_FLT_INEXACT_RESULT);
|
|
EXC(EXCEPTION_FLT_INVALID_OPERATION);
|
|
EXC(EXCEPTION_FLT_OVERFLOW);
|
|
EXC(EXCEPTION_FLT_STACK_CHECK);
|
|
EXC(EXCEPTION_FLT_UNDERFLOW);
|
|
EXC(EXCEPTION_ILLEGAL_INSTRUCTION);
|
|
EXC(EXCEPTION_IN_PAGE_ERROR);
|
|
EXC(EXCEPTION_INT_DIVIDE_BY_ZERO);
|
|
EXC(EXCEPTION_INT_OVERFLOW);
|
|
EXC(EXCEPTION_INVALID_DISPOSITION);
|
|
EXC(EXCEPTION_NONCONTINUABLE_EXCEPTION);
|
|
EXC(EXCEPTION_PRIV_INSTRUCTION);
|
|
EXC(EXCEPTION_SINGLE_STEP);
|
|
EXC(EXCEPTION_STACK_OVERFLOW);
|
|
#undef EXC
|
|
};
|
|
|
|
std::fprintf(stderr, "exception: (0x%x) %s caught:\n%s\n"
|
|
, code, name, stack_text);
|
|
|
|
output_test_log_to_terminal();
|
|
|
|
exit(code);
|
|
}
|
|
|
|
#else
|
|
|
|
void sig_handler(int sig)
|
|
{
|
|
char stack_text[10000];
|
|
|
|
#if (defined TORRENT_DEBUG && TORRENT_USE_ASSERTS) \
|
|
|| defined TORRENT_ASIO_DEBUGGING \
|
|
|| defined TORRENT_PROFILE_CALLS \
|
|
|| defined TORRENT_RELEASE_ASSERTS \
|
|
|| defined TORRENT_DEBUG_BUFFERS
|
|
print_backtrace(stack_text, sizeof(stack_text), 30);
|
|
#elif defined __FUNCTION__
|
|
strcat(stack_text, __FUNCTION__);
|
|
#else
|
|
stack_text[0] = 0;
|
|
strcat(stack_text, "<stack traces disabled>");
|
|
#endif
|
|
char const* sig_name = 0;
|
|
switch (sig)
|
|
{
|
|
#define SIG(x) case x: sig_name = #x; break
|
|
SIG(SIGSEGV);
|
|
#ifdef SIGBUS
|
|
SIG(SIGBUS);
|
|
#endif
|
|
SIG(SIGINT);
|
|
SIG(SIGTERM);
|
|
SIG(SIGILL);
|
|
SIG(SIGABRT);
|
|
SIG(SIGFPE);
|
|
#ifdef SIGSYS
|
|
SIG(SIGSYS);
|
|
#endif
|
|
#undef SIG
|
|
};
|
|
fprintf(stderr, "signal: %s caught:\n%s\n", sig_name, stack_text);
|
|
|
|
output_test_log_to_terminal();
|
|
|
|
#ifdef WIN32
|
|
exit(sig);
|
|
#else
|
|
exit(128 + sig);
|
|
#endif
|
|
}
|
|
|
|
#endif // _WIN32
|
|
|
|
void print_usage(char const* executable)
|
|
{
|
|
printf("%s [options] [tests...]\n"
|
|
"\n"
|
|
"OPTIONS:\n"
|
|
"-h,--help show this help\n"
|
|
"-l,--list list the tests available to run\n"
|
|
"-k,--keep keep files created by the test\n"
|
|
" regardless of whether it passed or not\n"
|
|
"-n,--no-redirect don't redirect test output to\n"
|
|
" temporary file, but let it go straight\n"
|
|
" to stdout\n"
|
|
"--no-stderr-redirect don't redirect stderr, but still redirect\n"
|
|
" stdout. This is useful when building with\n"
|
|
" sanitizers, which rely on being able to print\n"
|
|
" to stderr and exit\n"
|
|
"\n"
|
|
"for tests, specify one or more test names as printed\n"
|
|
"by -l. If no test is specified, all tests are run\n", executable);
|
|
}
|
|
|
|
void change_directory(std::string const& f, error_code& ec)
|
|
{
|
|
ec.clear();
|
|
|
|
#ifdef TORRENT_WINDOWS
|
|
#if TORRENT_USE_WSTRING
|
|
#define SetCurrentDirectory_ SetCurrentDirectoryW
|
|
std::wstring n = convert_to_wstring(f);
|
|
#else
|
|
#define SetCurrentDirectory_ SetCurrentDirectoryA
|
|
std::string const& n = convert_to_native(f);
|
|
#endif // TORRENT_USE_WSTRING
|
|
|
|
if (SetCurrentDirectory_(n.c_str()) == 0)
|
|
ec.assign(GetLastError(), system_category());
|
|
#else
|
|
std::string n = convert_to_native(f);
|
|
int ret = ::chdir(n.c_str());
|
|
if (ret != 0)
|
|
ec.assign(errno, system_category());
|
|
#endif
|
|
}
|
|
|
|
struct unit_directory_guard
|
|
{
|
|
std::string dir;
|
|
unit_directory_guard(std::string const& d) : dir(d) {}
|
|
~unit_directory_guard()
|
|
{
|
|
error_code ec;
|
|
std::string parent_dir = parent_path(dir);
|
|
change_directory(parent_dir, ec); // windows will not allow to remove current dir, so let's change it to root
|
|
if (ec)
|
|
{
|
|
TEST_ERROR("Failed to change directory: " + ec.message());
|
|
return;
|
|
}
|
|
if (!keep_files)
|
|
{
|
|
error_code ec;
|
|
remove_all(dir, ec);
|
|
if (ec)
|
|
TEST_ERROR("Failed to remove unit test directory: " + ec.message());
|
|
}
|
|
}
|
|
};
|
|
|
|
EXPORT int main(int argc, char const* argv[])
|
|
{
|
|
char const* executable = argv[0];
|
|
// skip executable name
|
|
++argv;
|
|
--argc;
|
|
|
|
// pick up options
|
|
while (argc > 0 && argv[0][0] == '-')
|
|
{
|
|
if (strcmp(argv[0], "-h") == 0 || strcmp(argv[0], "--help") == 0)
|
|
{
|
|
print_usage(executable);
|
|
return 0;
|
|
}
|
|
|
|
if (strcmp(argv[0], "-l") == 0 || strcmp(argv[0], "--list") == 0)
|
|
{
|
|
printf("TESTS:\n");
|
|
for (int i = 0; i < _g_num_unit_tests; ++i)
|
|
{
|
|
printf(" - %s\n", _g_unit_tests[i].name);
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
if (strcmp(argv[0], "-n") == 0 || strcmp(argv[0], "--no-redirect") == 0)
|
|
{
|
|
redirect_stdout = false;
|
|
redirect_stderr = false;
|
|
}
|
|
|
|
if (strcmp(argv[0], "--no-stderr-redirect") == 0)
|
|
{
|
|
redirect_stderr = false;
|
|
}
|
|
|
|
if (strcmp(argv[0], "-k") == 0 || strcmp(argv[0], "--keep") == 0)
|
|
{
|
|
keep_files = true;
|
|
}
|
|
++argv;
|
|
--argc;
|
|
}
|
|
|
|
std::set<std::string> tests_to_run;
|
|
bool filter = false;
|
|
|
|
for (int i = 0; i < argc; ++i)
|
|
{
|
|
tests_to_run.insert(argv[i]);
|
|
filter = true;
|
|
}
|
|
|
|
#ifdef O_NONBLOCK
|
|
// on darwin, stdout is set to non-blocking mode by default
|
|
// which sometimes causes tests to fail with EAGAIN just
|
|
// by printing logs
|
|
int flags = fcntl(fileno(stdout), F_GETFL, 0);
|
|
fcntl(fileno(stdout), F_SETFL, flags & ~O_NONBLOCK);
|
|
flags = fcntl(fileno(stderr), F_GETFL, 0);
|
|
fcntl(fileno(stderr), F_SETFL, flags & ~O_NONBLOCK);
|
|
#endif
|
|
|
|
#ifdef WIN32
|
|
// try to suppress hanging the process by windows displaying
|
|
// modal dialogs.
|
|
SetErrorMode( SEM_NOALIGNMENTFAULTEXCEPT
|
|
| SEM_NOGPFAULTERRORBOX | SEM_NOOPENFILEERRORBOX);
|
|
|
|
SetUnhandledExceptionFilter(&seh_exception_handler);
|
|
|
|
#ifdef _DEBUG
|
|
_CrtSetReportMode(_CRT_ERROR, _CRTDBG_MODE_FILE);
|
|
_CrtSetReportFile(_CRT_ERROR, _CRTDBG_FILE_STDERR);
|
|
#endif
|
|
|
|
#else
|
|
|
|
signal(SIGSEGV, &sig_handler);
|
|
#ifdef SIGBUS
|
|
signal(SIGBUS, &sig_handler);
|
|
#endif
|
|
signal(SIGILL, &sig_handler);
|
|
signal(SIGINT, &sig_handler);
|
|
signal(SIGABRT, &sig_handler);
|
|
signal(SIGFPE, &sig_handler);
|
|
#ifdef SIGSYS
|
|
signal(SIGSYS, &sig_handler);
|
|
#endif
|
|
|
|
#endif // WIN32
|
|
|
|
int process_id = -1;
|
|
#ifdef _WIN32
|
|
process_id = _getpid();
|
|
#else
|
|
process_id = getpid();
|
|
#endif
|
|
std::string root_dir = current_working_directory();
|
|
char dir[40];
|
|
snprintf(dir, sizeof(dir), "test_tmp_%u", process_id);
|
|
std::string unit_dir_prefix = combine_path(root_dir, dir);
|
|
std::printf("cwd_prefix = \"%s\"\n", unit_dir_prefix.c_str());
|
|
|
|
if (_g_num_unit_tests == 0)
|
|
{
|
|
fprintf(stderr, "\x1b[31mERROR: no unit tests registered\x1b[0m\n");
|
|
return 1;
|
|
}
|
|
|
|
if (redirect_stdout) old_stdout = dup(fileno(stdout));
|
|
if (redirect_stderr) old_stderr = dup(fileno(stderr));
|
|
|
|
int num_run = 0;
|
|
for (int i = 0; i < _g_num_unit_tests; ++i)
|
|
{
|
|
if (filter && tests_to_run.count(_g_unit_tests[i].name) == 0)
|
|
continue;
|
|
|
|
std::string unit_dir = unit_dir_prefix;
|
|
char i_str[40];
|
|
snprintf(i_str, sizeof(i_str), "%u", i);
|
|
unit_dir.append(i_str);
|
|
error_code ec;
|
|
create_directory(unit_dir, ec);
|
|
if (ec)
|
|
{
|
|
std::printf("Failed to create unit test directory: %s\n", ec.message().c_str());
|
|
return 1;
|
|
}
|
|
unit_directory_guard unit_dir_guard(unit_dir);
|
|
change_directory(unit_dir, ec);
|
|
if (ec)
|
|
{
|
|
std::printf("Failed to change unit test directory: %s\n", ec.message().c_str());
|
|
return 1;
|
|
}
|
|
|
|
unit_test_t& t = _g_unit_tests[i];
|
|
|
|
if (redirect_stdout)
|
|
{
|
|
// redirect test output to a temporary file
|
|
fflush(stdout);
|
|
fflush(stderr);
|
|
|
|
FILE* f = tmpfile();
|
|
if (f != NULL)
|
|
{
|
|
int ret1 = dup2(fileno(f), fileno(stdout));
|
|
if (redirect_stderr) dup2(fileno(f), fileno(stderr));
|
|
if (ret1 >= 0)
|
|
{
|
|
t.output = f;
|
|
}
|
|
else
|
|
{
|
|
fprintf(stderr, "failed to redirect output: (%d) %s\n"
|
|
, errno, strerror(errno));
|
|
}
|
|
}
|
|
else
|
|
{
|
|
fprintf(stderr, "failed to create temporary file for redirecting "
|
|
"output: (%d) %s\n", errno, strerror(errno));
|
|
}
|
|
}
|
|
|
|
// get proper interleaving of stderr and stdout
|
|
setbuf(stdout, NULL);
|
|
setbuf(stderr, NULL);
|
|
|
|
_g_test_idx = i;
|
|
current_test = &t;
|
|
|
|
#ifndef BOOST_NO_EXCEPTIONS
|
|
try
|
|
{
|
|
#endif
|
|
|
|
_g_test_failures = 0;
|
|
(*t.fun)();
|
|
#ifndef BOOST_NO_EXCEPTIONS
|
|
}
|
|
catch (std::exception const& e)
|
|
{
|
|
char buf[200];
|
|
snprintf(buf, sizeof(buf), "Terminated with exception: \"%s\"", e.what());
|
|
report_failure(buf, __FILE__, __LINE__);
|
|
}
|
|
catch (...)
|
|
{
|
|
report_failure("Terminated with unknown exception", __FILE__, __LINE__);
|
|
}
|
|
#endif
|
|
|
|
if (!tests_to_run.empty()) tests_to_run.erase(t.name);
|
|
|
|
if (_g_test_failures > 0)
|
|
{
|
|
output_test_log_to_terminal();
|
|
}
|
|
|
|
t.num_failures = _g_test_failures;
|
|
t.run = true;
|
|
++num_run;
|
|
|
|
if (redirect_stdout && t.output)
|
|
fclose(t.output);
|
|
}
|
|
|
|
if (redirect_stdout) dup2(old_stdout, fileno(stdout));
|
|
if (redirect_stderr) dup2(old_stderr, fileno(stderr));
|
|
|
|
if (!tests_to_run.empty())
|
|
{
|
|
fprintf(stderr, "\x1b[1mUNKONWN tests:\x1b[0m\n");
|
|
for (std::set<std::string>::iterator i = tests_to_run.begin()
|
|
, end(tests_to_run.end()); i != end; ++i)
|
|
{
|
|
fprintf(stderr, " %s\n", i->c_str());
|
|
}
|
|
}
|
|
|
|
if (num_run == 0)
|
|
{
|
|
fprintf(stderr, "\x1b[31mERROR: no unit tests run\x1b[0m\n");
|
|
return 1;
|
|
}
|
|
|
|
// just in case of premature exits
|
|
// make sure we try to clean up some
|
|
stop_udp_tracker();
|
|
stop_all_proxies();
|
|
stop_web_server();
|
|
stop_peer();
|
|
stop_dht();
|
|
|
|
if (redirect_stdout) fflush(stdout);
|
|
if (redirect_stderr) fflush(stderr);
|
|
|
|
int total_num_failures = print_failures();
|
|
|
|
return total_num_failures ? 333 : 0;
|
|
}
|
|
|