forked from premiere/premiere-libtorrent
488 lines
12 KiB
C++
488 lines
12 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 <cstdio>
|
|
#include <cstdlib> // 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 <boost/system/system_error.hpp>
|
|
|
|
#include "libtorrent/assert.hpp"
|
|
#include "libtorrent/file.hpp"
|
|
#include "libtorrent/random.hpp"
|
|
#include <csignal>
|
|
|
|
#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 = true;
|
|
bool keep_files = false;
|
|
|
|
extern int _g_test_idx;
|
|
|
|
// the current tests file descriptor
|
|
unit_test_t* current_test = nullptr;
|
|
|
|
void output_test_log_to_terminal()
|
|
{
|
|
if (current_test == nullptr || old_stdout == -1 || old_stderr == -1
|
|
|| !redirect_stdout || current_test->output == nullptr)
|
|
return;
|
|
|
|
fflush(stdout);
|
|
fflush(stderr);
|
|
dup2(old_stdout, fileno(stdout));
|
|
dup2(old_stderr, fileno(stderr));
|
|
|
|
fseek(current_test->output, 0, SEEK_SET);
|
|
std::printf("\x1b[1m[%s]\x1b[0m\n\n", current_test->name);
|
|
char buf[4096];
|
|
int size = 0;
|
|
do {
|
|
size = int(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__
|
|
strcpy(stack_text, __FUNCTION__);
|
|
#else
|
|
strcpy(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::printf("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 TORRENT_USE_ASSERTS \
|
|
|| defined TORRENT_ASIO_DEBUGGING \
|
|
|| defined TORRENT_PROFILE_CALLS \
|
|
|| defined TORRENT_DEBUG_BUFFERS
|
|
print_backtrace(stack_text, sizeof(stack_text), 30);
|
|
#elif defined __FUNCTION__
|
|
strcpy(stack_text, __FUNCTION__);
|
|
#else
|
|
strcpy(stack_text, "<stack traces disabled>");
|
|
#endif
|
|
char const* name = "<unknown signal>";
|
|
switch (sig)
|
|
{
|
|
#define SIG(x) case x: 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
|
|
};
|
|
std::printf("signal: (%d) %s caught:\n%s\n"
|
|
, sig, name, stack_text);
|
|
|
|
output_test_log_to_terminal();
|
|
|
|
exit(128 + sig);
|
|
}
|
|
|
|
#endif // _WIN32
|
|
|
|
void print_usage(char const* executable)
|
|
{
|
|
std::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);
|
|
}
|
|
|
|
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)
|
|
{
|
|
std::printf("TESTS:\n");
|
|
for (int i = 0; i < _g_num_unit_tests; ++i)
|
|
{
|
|
std::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
|
|
char dir[40];
|
|
std::snprintf(dir, sizeof(dir), "test_tmp_%u", process_id);
|
|
std::string test_dir = complete(dir);
|
|
error_code ec;
|
|
create_directory(test_dir, ec);
|
|
if (ec)
|
|
{
|
|
std::printf("Failed to create test directory: %s\n", ec.message().c_str());
|
|
return 1;
|
|
}
|
|
int ret;
|
|
#ifdef TORRENT_WINDOWS
|
|
SetCurrentDirectoryA(dir);
|
|
#else
|
|
ret = chdir(dir);
|
|
if (ret != 0)
|
|
{
|
|
std::printf("failed to change directory to \"%s\": %s"
|
|
, dir, strerror(errno));
|
|
return 1;
|
|
}
|
|
#endif
|
|
std::printf("test: %s\ncwd = \"%s\"\nrnd: %x\n"
|
|
, executable, test_dir.c_str(), libtorrent::random(0xffffffff));
|
|
|
|
int total_failures = 0;
|
|
|
|
if (_g_num_unit_tests == 0)
|
|
{
|
|
std::printf("\x1b[31mTEST_ERROR: 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;
|
|
|
|
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 != nullptr)
|
|
{
|
|
int ret1 = dup2(fileno(f), fileno(stdout));
|
|
if (redirect_stderr) dup2(fileno(f), fileno(stderr));
|
|
if (ret1 >= 0)
|
|
{
|
|
t.output = f;
|
|
}
|
|
else
|
|
{
|
|
std::printf("failed to redirect output: (%d) %s\n"
|
|
, errno, strerror(errno));
|
|
}
|
|
}
|
|
else
|
|
{
|
|
std::printf("failed to create temporary file for redirecting "
|
|
"output: (%d) %s\n", errno, strerror(errno));
|
|
}
|
|
}
|
|
|
|
// get proper interleaving of stderr and stdout
|
|
setbuf(stdout, nullptr);
|
|
setbuf(stderr, nullptr);
|
|
|
|
_g_test_idx = i;
|
|
current_test = &t;
|
|
|
|
#ifndef BOOST_NO_EXCEPTIONS
|
|
try
|
|
{
|
|
#endif
|
|
|
|
_g_test_failures = 0;
|
|
(*t.fun)();
|
|
#ifndef BOOST_NO_EXCEPTIONS
|
|
}
|
|
catch (boost::system::system_error const& e)
|
|
{
|
|
char buf[200];
|
|
std::snprintf(buf, sizeof(buf), "TEST_ERROR: Terminated with system_error: (%d) [%s] \"%s\""
|
|
, e.code().value()
|
|
, e.code().category().name()
|
|
, e.code().message().c_str());
|
|
report_failure(buf, __FILE__, __LINE__);
|
|
}
|
|
catch (std::exception const& e)
|
|
{
|
|
char buf[200];
|
|
std::snprintf(buf, sizeof(buf), "TEST_ERROR: Terminated with exception: \"%s\"", e.what());
|
|
report_failure(buf, __FILE__, __LINE__);
|
|
}
|
|
catch (...)
|
|
{
|
|
report_failure("TEST_ERROR: 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;
|
|
total_failures += _g_test_failures;
|
|
++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())
|
|
{
|
|
std::printf("\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)
|
|
{
|
|
std::printf(" %s\n", i->c_str());
|
|
}
|
|
}
|
|
|
|
if (num_run == 0)
|
|
{
|
|
std::printf("\x1b[31mTEST_ERROR: 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);
|
|
|
|
ret = print_failures();
|
|
#if !defined TORRENT_LOGGING
|
|
if (ret == 0 && !keep_files)
|
|
{
|
|
remove_all(test_dir, ec);
|
|
if (ec)
|
|
std::printf("failed to remove test dir: %s\n", ec.message().c_str());
|
|
}
|
|
#endif
|
|
|
|
return total_failures ? 333 : 0;
|
|
}
|
|
|