notcurses/src/tests/main.cpp

174 lines
4.7 KiB
C++
Raw Normal View History

#define DOCTEST_CONFIG_IMPLEMENT
#include "main.h"
#include <term.h>
#include <fcntl.h>
2019-11-17 10:04:41 +00:00
#include <clocale>
2019-12-27 22:43:24 +00:00
#include <cstring>
#include <cstdlib>
2019-11-17 10:04:41 +00:00
#include <iostream>
2020-05-07 04:53:01 +00:00
#include <climits>
#include <termios.h>
#include <sys/stat.h>
2020-05-07 04:53:01 +00:00
#include <filesystem>
#include <langinfo.h>
2019-11-17 10:04:41 +00:00
const char* datadir = NOTCURSES_SHARE;
ncloglevel_e loglevel = NCLOGLEVEL_SILENT;
2019-12-27 22:43:24 +00:00
auto testing_notcurses() -> struct notcurses* {
notcurses_options nopts{};
// get loglevel from command line. enabling it by default leads to
// more confusion than useful information, so leave it off by default.
nopts.loglevel = loglevel;
2020-12-31 05:32:47 +00:00
nopts.flags = NCOPTION_SUPPRESS_BANNERS | NCOPTION_NO_ALTERNATE_SCREEN;
auto nc = notcurses_init(&nopts, nullptr);
return nc;
}
2020-05-07 04:53:01 +00:00
auto find_data(const char* datum) -> char* {
std::filesystem::path p = datadir;
p /= datum;
return strdup(p.c_str());
}
2021-05-17 20:19:13 +00:00
auto is_test_tty(FILE* fp) -> bool {
int fd = fileno(fp);
if(fd >= 0){
if(isatty(fd)){
return true;
}
}
return false;
}
2019-12-27 22:43:24 +00:00
static void
handle_opts(const char** argv){
bool inarg = false;
while(*argv){
if(inarg){
datadir = strdup(*argv);
2019-12-27 22:43:24 +00:00
inarg = false;
}else if(strcmp(*argv, "-p") == 0){
inarg = true;
2021-05-16 11:33:24 +00:00
}else if(strncmp(*argv, "-l", 2) == 0){ // just require -l
loglevel = NCLOGLEVEL_TRACE;
2019-12-27 22:43:24 +00:00
}
++argv;
}
}
// check that the (provided or default) data directory exists, and has at
// least one of our necessary files. otherwise, print a warning + return error.
static int
check_data_dir(){
auto p = find_data("changes.jpg");
if(!p){
std::cerr << "Coudln't find testing data! Supply directory with -p." << std::endl;
return -1;
}
struct stat s;
if(stat(p, &s)){
std::cerr << "Couldn't open " << p << ". Supply directory with -p." << std::endl;
free(p);
return -1;
}
free(p);
return 0;
}
// reset the terminal in the event of early exit (notcurses_init() presumably
// ran, but we don't have the notcurses struct to destroy, so just do it raw).
static void
reset_terminal(){
int fd = open("/dev/tty", O_RDWR|O_CLOEXEC);
if(fd >= 0){
struct termios tios;
if(tcgetattr(fd, &tios) == 0){
tios.c_iflag |= INLCR;
tios.c_lflag |= ISIG | ICANON | ECHO;
tcsetattr(fd, TCSADRAIN, &tios);
}
char* str = tigetstr("sgr0");
if(str != (char*)-1){
printf("%s", str);
}
fflush(stdout);
str = tigetstr("oc");
if(str != (char*)-1){
printf("%s", str);
}
fflush(stdout);
str = tigetstr("cnorm");
if(str != (char*)-1){
printf("%s", str);
}
fflush(stdout);
close(fd);
}
}
2019-12-27 22:43:24 +00:00
// from https://github.com/onqtam/doctest/blob/master/doc/markdown/commandline.md
class dt_removed {
std::vector<const char*> vec;
public:
dt_removed(const char** argv_in) {
for(; *argv_in; ++argv_in)
if(strncmp(*argv_in, "--dt-", strlen("--dt-")) != 0)
vec.push_back(*argv_in);
2020-05-07 04:53:01 +00:00
vec.push_back(nullptr);
2019-12-27 22:43:24 +00:00
}
2020-05-07 04:53:01 +00:00
auto argc() -> int { return static_cast<int>(vec.size()) - 1; }
auto argv() -> const char** { return &vec[0]; }
2019-12-27 22:43:24 +00:00
};
auto lang_and_term() -> void {
const char* lang = getenv("LANG");
if(lang == nullptr){
std::cerr << "Warning: LANG wasn't defined" << std::endl;
}else{
std::cout << "Running with LANG=" << lang << std::endl;
}
const char* term = getenv("TERM");
// ubuntu's buildd sets TERM=unknown, fuck it, handle this atrocity
if(term == nullptr || strcmp(term, "unknown") == 0){
std::cerr << "TERM wasn't defined, exiting with success" << std::endl;
exit(EXIT_SUCCESS);
}
auto nc = testing_notcurses();
if(!nc){
exit(EXIT_FAILURE);
}
int dimy, dimx;
notcurses_stddim_yx(nc, &dimy, &dimx);
std::cout << "Running with TERM=" << term << " (" << dimx << 'x'
<< dimy << ")" << std::endl;
notcurses_stop(nc);
if(dimx < 80 || dimy < 24){ // minimum assumed geometry
std::cerr << "Terminal was too small for tests (minimum 80x24)" << std::endl;
exit(EXIT_FAILURE);
}
}
auto main(int argc, const char **argv) -> int {
if(!setlocale(LC_ALL, "")){
std::cerr << "Couldn't set locale based on user preferences!" << std::endl;
return EXIT_FAILURE;
}
lang_and_term(); // might exit
doctest::Context context;
context.setOption("order-by", "name"); // sort the test cases by their name
context.applyCommandLine(argc, argv);
context.setOption("no-breaks", true); // don't break in the debugger when assertions fail
2019-12-27 22:43:24 +00:00
dt_removed args(argv);
handle_opts(argv);
int res = context.run(); // run
reset_terminal();
check_data_dir();
if(context.shouldExit()){ // important - query flags (and --exit) rely on the user doing this
return res; // propagate the result of the tests
}
return res; // the result from doctest is propagated here as well
2019-11-17 10:04:41 +00:00
}