mirror of
https://github.com/autc04/Retro68.git
synced 2025-01-11 02:30:42 +00:00
Automated tests!
This commit is contained in:
parent
8b5f06cb9b
commit
360a9858e3
20
AutomatedTests/CMakeLists.txt
Normal file
20
AutomatedTests/CMakeLists.txt
Normal file
@ -0,0 +1,20 @@
|
||||
enable_testing()
|
||||
|
||||
add_application(ReallyEmpty ReallyEmpty.c)
|
||||
add_test(NAME ReallyEmpty COMMAND LaunchAPPL -e ReallyEmpty.bin)
|
||||
|
||||
add_application(Empty Empty.c)
|
||||
add_test(NAME Empty COMMAND LaunchAPPL -e Empty.bin)
|
||||
|
||||
add_application(File File.c)
|
||||
add_test(NAME File
|
||||
COMMAND LaunchAPPL -e File.bin --logfile=out -t5
|
||||
)
|
||||
set_tests_properties(File PROPERTIES PASS_REGULAR_EXPRESSION "OK")
|
||||
|
||||
add_application(Timeout Timeout.c)
|
||||
add_test(NAME Timeout
|
||||
COMMAND sh -c "! LaunchAPPL -e Timeout.bin --logfile=out -t2"
|
||||
)
|
||||
set_tests_properties(Timeout PROPERTIES PASS_REGULAR_EXPRESSION "One;Two"
|
||||
FAIL_REGULAR_EXPRESSION "Three")
|
4
AutomatedTests/Empty.c
Normal file
4
AutomatedTests/Empty.c
Normal file
@ -0,0 +1,4 @@
|
||||
int main()
|
||||
{
|
||||
return 0;
|
||||
}
|
6
AutomatedTests/File.c
Normal file
6
AutomatedTests/File.c
Normal file
@ -0,0 +1,6 @@
|
||||
#include "Test.h"
|
||||
|
||||
int main()
|
||||
{
|
||||
TEST_LOG_SIZED("OK", 2);
|
||||
}
|
4
AutomatedTests/ReallyEmpty.c
Normal file
4
AutomatedTests/ReallyEmpty.c
Normal file
@ -0,0 +1,4 @@
|
||||
void _start()
|
||||
{
|
||||
}
|
||||
|
6
AutomatedTests/Test.c
Normal file
6
AutomatedTests/Test.c
Normal file
@ -0,0 +1,6 @@
|
||||
#include "Test.h"
|
||||
|
||||
void TestLog(const char *str)
|
||||
{
|
||||
TEST_LOG_SIZED(str, strlen(str));
|
||||
}
|
40
AutomatedTests/Test.h
Normal file
40
AutomatedTests/Test.h
Normal file
@ -0,0 +1,40 @@
|
||||
#ifndef TEST_H
|
||||
#define TEST_H
|
||||
|
||||
#include <Files.h>
|
||||
#include <Devices.h>
|
||||
|
||||
#define TEST_LOG_SIZED(str, size) \
|
||||
do { \
|
||||
HParamBlockRec _hpb; \
|
||||
\
|
||||
unsigned char _fileName[4]; \
|
||||
_fileName[0] = 3; \
|
||||
_fileName[1] = 'o'; \
|
||||
_fileName[2] = 'u'; \
|
||||
_fileName[3] = 't'; \
|
||||
\
|
||||
_hpb.ioParam.ioCompletion = NULL; \
|
||||
_hpb.ioParam.ioNamePtr = (StringPtr)_fileName; \
|
||||
_hpb.ioParam.ioVRefNum = 0; \
|
||||
_hpb.fileParam.ioDirID = 0; \
|
||||
_hpb.ioParam.ioPermssn = fsRdWrPerm; \
|
||||
PBHOpenSync(&_hpb); \
|
||||
\
|
||||
_hpb.ioParam.ioBuffer = str; \
|
||||
_hpb.ioParam.ioReqCount = size; \
|
||||
_hpb.ioParam.ioPosMode = fsFromLEOF; \
|
||||
_hpb.ioParam.ioPosOffset = 0; \
|
||||
PBWriteSync((void*)&_hpb); \
|
||||
char _newline = '\n'; \
|
||||
_hpb.ioParam.ioBuffer = &_newline; \
|
||||
_hpb.ioParam.ioReqCount = 1; \
|
||||
_hpb.ioParam.ioPosMode = fsFromLEOF; \
|
||||
_hpb.ioParam.ioPosOffset = 0; \
|
||||
PBWriteSync((void*)&_hpb); \
|
||||
PBCloseSync((void*)&_hpb); \
|
||||
} while(0);
|
||||
|
||||
void TestLog(const char *str);
|
||||
|
||||
#endif // TEST_H
|
11
AutomatedTests/Timeout.c
Normal file
11
AutomatedTests/Timeout.c
Normal file
@ -0,0 +1,11 @@
|
||||
#include "Test.h"
|
||||
|
||||
int main()
|
||||
{
|
||||
TEST_LOG_SIZED("One",3);
|
||||
TEST_LOG_SIZED("Two",3);
|
||||
for(;;)
|
||||
;
|
||||
TEST_LOG_SIZED("Three",5);
|
||||
return 0;
|
||||
}
|
@ -48,6 +48,9 @@ add_subdirectory(Samples/Launcher)
|
||||
add_subdirectory(Samples/SystemExtension)
|
||||
endif()
|
||||
|
||||
enable_testing()
|
||||
add_subdirectory(AutomatedTests)
|
||||
|
||||
else()
|
||||
|
||||
set(RETRO68_ROOT ${CMAKE_INSTALL_PREFIX})
|
||||
@ -62,4 +65,5 @@ add_subdirectory(Rez)
|
||||
add_subdirectory(ConvertObj)
|
||||
add_subdirectory(PEFTools)
|
||||
add_subdirectory(Elf2Mac)
|
||||
add_subdirectory(LaunchAPPL)
|
||||
endif()
|
||||
|
7
LaunchAPPL/CMakeLists.txt
Normal file
7
LaunchAPPL/CMakeLists.txt
Normal file
@ -0,0 +1,7 @@
|
||||
find_package(Boost COMPONENTS filesystem program_options)
|
||||
|
||||
add_executable(LaunchAPPL LaunchAPPL.cc)
|
||||
target_include_directories(LaunchAPPL PRIVATE ${CMAKE_INSTALL_PREFIX}/include ${Boost_INCLUDE_DIR})
|
||||
target_link_libraries(LaunchAPPL ResourceFiles ${Boost_LIBRARIES})
|
||||
|
||||
install(TARGETS LaunchAPPL RUNTIME DESTINATION bin)
|
215
LaunchAPPL/LaunchAPPL.cc
Normal file
215
LaunchAPPL/LaunchAPPL.cc
Normal file
@ -0,0 +1,215 @@
|
||||
#include <boost/program_options.hpp>
|
||||
#include <boost/filesystem.hpp>
|
||||
#include <boost/filesystem/fstream.hpp>
|
||||
#include <iostream>
|
||||
#include <vector>
|
||||
#include <string>
|
||||
|
||||
#include <unistd.h>
|
||||
#include <sys/types.h>
|
||||
#include <sys/wait.h>
|
||||
|
||||
|
||||
#include "ResourceFork.h"
|
||||
#include "ResourceFile.h"
|
||||
|
||||
namespace po = boost::program_options;
|
||||
namespace fs = boost::filesystem;
|
||||
|
||||
using std::string;
|
||||
using std::vector;
|
||||
|
||||
static po::options_description desc;
|
||||
po::variables_map options;
|
||||
|
||||
static void usage()
|
||||
{
|
||||
std::cerr << "Usage: " << "LaunchAPPL [options] appl-file\n";
|
||||
std::cerr << desc << std::endl;
|
||||
}
|
||||
|
||||
|
||||
int ChildProcess(string program, vector<string> args)
|
||||
{
|
||||
std::vector<const char*> argv;
|
||||
argv.push_back(program.c_str());
|
||||
for(string& s : args)
|
||||
argv.push_back(s.c_str());
|
||||
argv.push_back(NULL);
|
||||
|
||||
int timeout = options.count("timeout") ? options["timeout"].as<int>() : 0;
|
||||
|
||||
pid_t pid = fork();
|
||||
if(pid < 0)
|
||||
{
|
||||
perror("unable to fork");
|
||||
return 1;
|
||||
}
|
||||
else if(pid == 0)
|
||||
{
|
||||
pid_t worker_pid = timeout ? fork() : 0;
|
||||
if(worker_pid < 0)
|
||||
{
|
||||
perror("unable to fork");
|
||||
_exit(1);
|
||||
}
|
||||
if(worker_pid == 0)
|
||||
{
|
||||
execvp(argv[0], const_cast<char* const *> (argv.data()));
|
||||
perror("exec failed");
|
||||
_exit(1);
|
||||
}
|
||||
|
||||
pid_t timeout_pid = fork();
|
||||
if(timeout_pid < 0)
|
||||
{
|
||||
perror("unable to fork");
|
||||
_exit(1);
|
||||
}
|
||||
if(timeout_pid == 0)
|
||||
{
|
||||
sleep(timeout);
|
||||
_exit(0);
|
||||
}
|
||||
int wstatus;
|
||||
pid_t exited_pid = wait(&wstatus);
|
||||
if(exited_pid == worker_pid)
|
||||
{
|
||||
kill(timeout_pid, SIGKILL);
|
||||
wait(NULL);
|
||||
if(!WIFEXITED(wstatus))
|
||||
{
|
||||
return 1;
|
||||
}
|
||||
else
|
||||
{
|
||||
int exitcode = WEXITSTATUS(wstatus);
|
||||
_exit(exitcode);
|
||||
}
|
||||
}
|
||||
else
|
||||
{
|
||||
kill(worker_pid, SIGKILL);
|
||||
wait(NULL);
|
||||
_exit(1);
|
||||
}
|
||||
}
|
||||
else
|
||||
{
|
||||
int wstatus;
|
||||
int result = 0;
|
||||
do
|
||||
{
|
||||
result = waitpid(pid, &wstatus, 0);
|
||||
} while(result == -1 && errno == EINTR);
|
||||
|
||||
if(!WIFEXITED(wstatus))
|
||||
{
|
||||
return 1;
|
||||
}
|
||||
else
|
||||
{
|
||||
int exitcode = WEXITSTATUS(wstatus);
|
||||
return exitcode;
|
||||
}
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
int main(int argc, char *argv[])
|
||||
{
|
||||
desc.add_options()
|
||||
("help,h", "show this help message")
|
||||
("executor-path", po::value<std::string>()->default_value("executor"),"path to executor")
|
||||
("executor,e", "run using executor")
|
||||
("timeout,t", po::value<int>(),"abort after timeout")
|
||||
("logfile", po::value<std::string>(), "read log file")
|
||||
("result,r", "TEST 128")
|
||||
;
|
||||
po::options_description hidden, alldesc;
|
||||
hidden.add_options()
|
||||
("application,a", po::value<std::string>(), "application" )
|
||||
;
|
||||
alldesc.add(desc).add(hidden);
|
||||
|
||||
try
|
||||
{
|
||||
auto parsed = po::command_line_parser(argc, argv)
|
||||
.options(alldesc)
|
||||
.positional(po::positional_options_description().add("application", -1))
|
||||
.style(po::command_line_style::default_style)
|
||||
.run();
|
||||
|
||||
po::store(parsed, options);
|
||||
}
|
||||
catch(po::error& e)
|
||||
{
|
||||
std::cerr << "ERROR: " << e.what() << std::endl << std::endl;
|
||||
usage();
|
||||
return 1;
|
||||
}
|
||||
|
||||
po::notify(options);
|
||||
|
||||
if(options.count("help") || !options.count("application"))
|
||||
{
|
||||
usage();
|
||||
return 0;
|
||||
}
|
||||
|
||||
ResourceFile app(options["application"].as<std::string>());
|
||||
if(!app.read())
|
||||
{
|
||||
std::cerr << "Could not read application file.\n";
|
||||
return 1;
|
||||
}
|
||||
|
||||
|
||||
|
||||
if(options.count("executor"))
|
||||
{
|
||||
fs::path tempDir = fs::unique_path();
|
||||
std::cerr << "Unique path: " << tempDir.string() << std::endl;
|
||||
fs::create_directories(tempDir);
|
||||
|
||||
fs::path appPath = tempDir / "Application";
|
||||
|
||||
app.assign(appPath.string(), ResourceFile::Format::percent_appledouble);
|
||||
if(!app.write())
|
||||
{
|
||||
std::cerr << "Could not write application file.\n";
|
||||
return 1;
|
||||
}
|
||||
|
||||
if(options.count("logfile"))
|
||||
{
|
||||
fs::ofstream out(tempDir/options["logfile"].as<std::string>());
|
||||
}
|
||||
|
||||
int result = ChildProcess(options["executor-path"].as<std::string>(), { appPath.string() });
|
||||
|
||||
if(options.count("logfile"))
|
||||
{
|
||||
fs::ifstream in(tempDir/options["logfile"].as<std::string>());
|
||||
std::cout << in.rdbuf();
|
||||
}
|
||||
|
||||
|
||||
if(result == 0 && options.count("result"))
|
||||
{
|
||||
app.read();
|
||||
auto& resmap = app.resources.resources;
|
||||
auto p = resmap.find(ResRef("TEST", 128));
|
||||
if(p == resmap.end())
|
||||
return 1;
|
||||
|
||||
std::cout << p->second.getData();
|
||||
}
|
||||
|
||||
fs::remove_all(tempDir);
|
||||
|
||||
return result;
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
Loading…
x
Reference in New Issue
Block a user