diff --git a/.github/workflows/ci.yml b/.github/workflows/ci.yml index 9425a20..c7b2b4d 100644 --- a/.github/workflows/ci.yml +++ b/.github/workflows/ci.yml @@ -146,12 +146,12 @@ jobs: cd tests bash ./run.sh ${{ matrix.compiler }} v${{ matrix.version }} 32 - tests-macos11-gpp: + tests-macos12-gpp: strategy: matrix: - os: [macos-11] + os: [macos-12] compiler: [g++] - version: [10, 11, 12] + version: [12, 13, 14] name: Use ${{ matrix.compiler }}-${{ matrix.version }} on ${{ matrix.os }} runs-on: ${{ matrix.os }} steps: @@ -161,10 +161,10 @@ jobs: cd tests bash ./run.sh ${{ matrix.compiler }} v${{ matrix.version }} - tests-macos11-clang: + tests-macos12-clang: strategy: matrix: - os: [macos-11] + os: [macos-12] compiler: [clang++] name: Use ${{ matrix.compiler }} on ${{ matrix.os }} runs-on: ${{ matrix.os }} @@ -175,12 +175,12 @@ jobs: cd tests bash ./run.sh ${{ matrix.compiler }} - tests-macos12-gpp: + tests-macos13-gpp: strategy: matrix: - os: [macos-12] + os: [macos-13] compiler: [g++] - version: [11, 12, 13] + version: [12, 13, 14] name: Use ${{ matrix.compiler }}-${{ matrix.version }} on ${{ matrix.os }} runs-on: ${{ matrix.os }} steps: @@ -190,10 +190,10 @@ jobs: cd tests bash ./run.sh ${{ matrix.compiler }} v${{ matrix.version }} - tests-macos12-clang: + tests-macos13-clang: strategy: matrix: - os: [macos-12] + os: [macos-13] compiler: [clang++] name: Use ${{ matrix.compiler }} on ${{ matrix.os }} runs-on: ${{ matrix.os }} @@ -204,12 +204,12 @@ jobs: cd tests bash ./run.sh ${{ matrix.compiler }} - tests-macos13-gpp: + tests-macos14-gpp: strategy: matrix: - os: [macos-13] + os: [macos-14] compiler: [g++] - version: [11, 12, 13] + version: [12, 13, 14] name: Use ${{ matrix.compiler }}-${{ matrix.version }} on ${{ matrix.os }} runs-on: ${{ matrix.os }} steps: @@ -219,10 +219,10 @@ jobs: cd tests bash ./run.sh ${{ matrix.compiler }} v${{ matrix.version }} - tests-macos13-clang: + tests-macos14-clang: strategy: matrix: - os: [macos-13] + os: [macos-14] compiler: [clang++] name: Use ${{ matrix.compiler }} on ${{ matrix.os }} runs-on: ${{ matrix.os }} diff --git a/docs/read.me b/docs/read.me index 2b5d32f..b3c5a84 100644 --- a/docs/read.me +++ b/docs/read.me @@ -1,19 +1,107 @@ -TESTLIB - -== What is it? == -Testlib is simple library which helps you to write - * checkers - * validators - * generators - * interactors -for programming competitions problems. -You can find latest release of the library on https://github.com/MikeMirzayanov/testlib/ -Problem development management system Polygon completely supports testlib. - -== How to use? == -Easest way is to read c++ sources in the checkers/, validators/, generators/ and interactors/ folders. -Also some classes and methods in testlib have documentation. - -Thanks for using testlib, -Mike Mirzayanov - +# Testlib + +## Intro + +This project contains a C++ implementation of testlib. It is already being used in many programming contests in Russia, such as the Russian National Olympiad in Informatics and different stages of ICPC. Join! + +The library's C++ code is tested for compatibility with standard C++11 and higher on different versions of `g++`, `clang++`, and Microsoft Visual C++. + +This code has been used many times in Codeforces contests. + +## Samples + +### Checker + +This sample checker expects the same integer in the output and the answer. It ignores all white-spaces. See more examples in the package. + +```c++ +#include "testlib.h" + +int main(int argc, char * argv[]) { + setName("compares two signed integers"); + registerTestlibCmd(argc, argv); + int ja = ans.readInt(); + int pa = ouf.readInt(); + if (ja != pa) + quitf(_wa, "expected %d, found %d", ja, pa); + quitf(_ok, "answer is %d", ja); +} +``` + +### Interactor + +This sample interactor reads pairs of numbers from the input file, sends them to another program, reads +the result, and writes it to an output file (to be verified later). Another option could be to terminate +the interactor with `quitf(_wa, )`. + +```c++ +#include "testlib.h" +#include + +using namespace std; + +int main(int argc, char* argv[]) { + setName("Interactor A+B"); + registerInteraction(argc, argv); + + // reads number of queries from test (input) file + int n = inf.readInt(); + for (int i = 0; i < n; i++) { + // reads query from test (input) file + int a = inf.readInt(); + int b = inf.readInt(); + + // writes query to the solution, endl makes flush + cout << a << " " << b << endl; + + // writes output file to be verified by checker later + tout << ouf.readInt() << endl; + } + + // just message + quitf(_ok, "%d queries processed", n); +} +``` + +### Validator + +This code reads input from the standard input and checks that it contains only one integer between 1 and 100, inclusive. It also validates that the file ends with EOLN and EOF. On Windows, it expects #13#10 as EOLN, and it expects #10 as EOLN on other platforms. It does not ignore white-spaces, so it works very strictly. It will return a non-zero code in the case of illegal input and write a message to the standard output. See more examples in the package. + +```c++ +#include "testlib.h" + +int main(int argc, char* argv[]) { + registerValidation(argc, argv); + inf.readInt(1, 100, "n"); + inf.readEoln(); + inf.readEof(); +} +``` + +### Generator + +This generator outputs a random token to the standard output, containing Latin letters or digits. The length of the token will be between 1 and 1000, inclusive. It will use a uniformly distributed random generator. To generate different values, call it with different command-line parameters. It is typical behavior for a testlib generator to set up randseed by command line. See more examples in the package. + +```c++ +#include "testlib.h" + +int main(int argc, char* argv[]) { + registerGen(argc, argv, 1); + println(rnd.next(1, 10)); /* Random number in the range [1,10]. */ + println(rnd.next("[a-zA-Z0-9]{1,1000}")); /* Random word of length [1,1000]. */ +} +``` + +This generator outputs a random permutation; the size is equal to the first command-line argument. + +```c++ +#include "testlib.h" + +int main(int argc, char* argv[]) { + registerGen(argc, argv, 1); + + int n = opt(1); + println(n); + println(rnd.perm(n, 1)); +} +``` diff --git a/testlib.h b/testlib.h index d188bf1..4e1d368 100644 --- a/testlib.h +++ b/testlib.h @@ -25,7 +25,7 @@ * Copyright (c) 2005-2024 */ -#define VERSION "0.9.43" +#define VERSION "0.9.44" /* * Mike Mirzayanov @@ -330,6 +330,12 @@ static int __testlib_format_buffer_usage_count = 0; result = std::string(__testlib_format_buffer); \ __testlib_format_buffer_usage_count--; \ +#ifdef __GNUC__ +__attribute__ ((format (printf, 1, 2))) +#endif +std::string testlib_format_(const char *fmt, ...); +std::string testlib_format_(const std::string fmt, ...); + const long long __TESTLIB_LONGLONG_MAX = 9223372036854775807LL; const int __TESTLIB_MAX_TEST_CASE = 1073741823; @@ -430,19 +436,6 @@ inline std::string lowerCase(std::string s) { return s; } -#ifdef __GNUC__ -__attribute__ ((format (printf, 1, 2))) -#endif -std::string format(const char *fmt, ...) { - FMT_TO_RESULT(fmt, fmt, result); - return result; -} - -std::string format(const std::string fmt, ...) { - FMT_TO_RESULT(fmt, fmt.c_str(), result); - return result; -} - #ifdef __GNUC__ __attribute__((const)) #endif @@ -673,6 +666,30 @@ static std::string toString(const T &t) { void prepareOpts(int argc, char* argv[]); #endif +FILE* testlib_fopen_(const char* path, const char* mode) { +#ifdef _MSC_VER + FILE* result = NULL; + if (fopen_s(&result, path, mode) != 0) + return NULL; + else + return result; +#else + return std::fopen(path, mode); +#endif +} + +FILE* testlib_freopen_(const char* path, const char* mode, FILE* file) { +#ifdef _MSC_VER + FILE* result = NULL; + if (freopen_s(&result, path, mode, file) != 0) + return NULL; + else + return result; +#else + return std::freopen(path, mode, file); +#endif +} + /* * Very simple regex-like pattern. * It used for two purposes: validation and generation. @@ -1423,7 +1440,11 @@ static void __pattern_scanCounts(const std::string &s, size_t &pos, int &from, i if (parts[i].length() == 0) __testlib_fail("pattern: Illegal pattern (or part) \"" + s + "\""); int number; +#ifdef _MSC_VER + if (sscanf_s(parts[i].c_str(), "%d", &number) != 1) +#else if (std::sscanf(parts[i].c_str(), "%d", &number) != 1) +#endif __testlib_fail("pattern: Illegal pattern (or part) \"" + s + "\""); numbers.push_back(number); } @@ -1805,7 +1826,7 @@ class FileInputStreamReader : public InputStreamReader { void setTestCase(int testCase) { if (testCase < 0 || testCase > __TESTLIB_MAX_TEST_CASE) - __testlib_fail(format("testCase expected fit in [1,%d], but %d doesn't", __TESTLIB_MAX_TEST_CASE, testCase)); + __testlib_fail(testlib_format_("testCase expected fit in [1,%d], but %d doesn't", __TESTLIB_MAX_TEST_CASE, testCase)); readChars.push_back(testCase + 256); } @@ -2639,7 +2660,7 @@ class Validator { else if (fileName == "stderr") f = stderr, standard_file = true; else { - f = fopen(fileName.c_str(), "wb"); + f = testlib_fopen_(fileName.c_str(), "wb"); if (NULL == f) __testlib_fail("Validator::writeTestOverviewLog: can't write test overview log to (" + fileName + ")"); } @@ -2682,7 +2703,7 @@ class Validator { else if (_testMarkupFileName == "stderr") f = stderr, standard_file = true; else { - f = fopen(_testMarkupFileName.c_str(), "wb"); + f = testlib_fopen_(_testMarkupFileName.c_str(), "wb"); if (NULL == f) __testlib_fail("Validator::writeTestMarkup: can't write test markup to (" + _testMarkupFileName + ")"); } @@ -2729,7 +2750,7 @@ class Validator { else if (_testCaseFileName == "stderr") f = stderr, standard_file = true; else { - f = fopen(_testCaseFileName.c_str(), "wb"); + f = testlib_fopen_(_testCaseFileName.c_str(), "wb"); if (NULL == f) __testlib_fail("Validator::writeTestCase: can't write test case to (" + _testCaseFileName + ")"); } @@ -3127,7 +3148,7 @@ NORETURN void InStream::quit(TResult result, const char *msg) { break; default: if (result >= _partially) { - errorName = format("partially correct (%d) ", pctype); + errorName = testlib_format_("partially correct (%d) ", pctype); isPartial = true; quitscrS(LightYellow, errorName); } else @@ -3135,7 +3156,7 @@ NORETURN void InStream::quit(TResult result, const char *msg) { } if (resultName != "") { - resultFile = std::fopen(resultName.c_str(), "w"); + resultFile = testlib_fopen_(resultName.c_str(), "w"); if (resultFile == NULL) { resultName = ""; quit(_fail, "Can not write to the result file"); @@ -3151,7 +3172,7 @@ NORETURN void InStream::quit(TResult result, const char *msg) { else { if (__testlib_points == std::numeric_limits::infinity()) quit(_fail, "Expected points, but infinity found"); - std::string stringPoints = removeDoubleTrailingZeroes(format("%.10f", __testlib_points)); + std::string stringPoints = removeDoubleTrailingZeroes(testlib_format_("%.10f", __testlib_points)); std::fprintf(resultFile, "", outcomes[(int) result].c_str(), stringPoints.c_str()); } @@ -3252,7 +3273,7 @@ void InStream::reset(std::FILE *file) { close(); if (!stdfile && NULL == file) - if (NULL == (file = std::fopen(name.c_str(), "rb"))) { + if (NULL == (file = testlib_fopen_(name.c_str(), "rb"))) { if (mode == _output) quits(_pe, std::string("Output file not found: \"") + name + "\""); @@ -3483,9 +3504,9 @@ std::string InStream::readWord(const pattern &p, const std::string &variableName std::vector InStream::readWords(int size, const pattern &p, const std::string &variablesName, int indexBase) { - __testlib_readMany(readWords, readWord(p, variablesName), std::string, true); if (strict && !variablesName.empty()) validator.addVariable(variablesName); + __testlib_readMany(readWords, readWord(p, variablesName), std::string, true); } std::vector InStream::readWords(int size, int indexBase) { @@ -3499,9 +3520,9 @@ std::string InStream::readWord(const std::string &ptrn, const std::string &varia std::vector InStream::readWords(int size, const std::string &ptrn, const std::string &variablesName, int indexBase) { pattern p(ptrn); - __testlib_readMany(readWords, readWord(p, variablesName), std::string, true); if (strict && !variablesName.empty()) validator.addVariable(variablesName); + __testlib_readMany(readWords, readWord(p, variablesName), std::string, true); } std::string InStream::readToken(const pattern &p, const std::string &variableName) { @@ -3510,9 +3531,9 @@ std::string InStream::readToken(const pattern &p, const std::string &variableNam std::vector InStream::readTokens(int size, const pattern &p, const std::string &variablesName, int indexBase) { - __testlib_readMany(readTokens, readToken(p, variablesName), std::string, true); if (strict && !variablesName.empty()) validator.addVariable(variablesName); + __testlib_readMany(readTokens, readToken(p, variablesName), std::string, true); } std::vector InStream::readTokens(int size, int indexBase) { @@ -3526,9 +3547,9 @@ std::string InStream::readToken(const std::string &ptrn, const std::string &vari std::vector InStream::readTokens(int size, const std::string &ptrn, const std::string &variablesName, int indexBase) { pattern p(ptrn); - __testlib_readMany(readTokens, readWord(p, variablesName), std::string, true); if (strict && !variablesName.empty()) validator.addVariable(variablesName); + __testlib_readMany(readTokens, readWord(p, variablesName), std::string, true); } void InStream::readWordTo(std::string &result, const pattern &p, const std::string &variableName) { @@ -3657,7 +3678,12 @@ static inline double stringToDouble(InStream &in, const char *buffer) { char *suffix = new char[length + 1]; std::memset(suffix, 0, length + 1); - int scanned = std::sscanf(buffer, "%lf%s", &result, suffix); + int scanned; +#ifdef _MSC_VER + scanned = sscanf_s(buffer, "%lf%s", &result, suffix, (unsigned int)(length + 1)); +#else + scanned = std::sscanf(buffer, "%lf%s", &result, suffix); +#endif bool empty = strlen(suffix) == 0; delete[] suffix; @@ -3734,7 +3760,12 @@ static inline double stringToStrictDouble(InStream &in, const char *buffer, char *suffix = new char[length + 1]; std::memset(suffix, 0, length + 1); - int scanned = std::sscanf(buffer, "%lf%s", &result, suffix); + int scanned; +#ifdef _MSC_VER + scanned = sscanf_s(buffer, "%lf%s", &result, suffix, (unsigned int)(length + 1)); +#else + scanned = std::sscanf(buffer, "%lf%s", &result, suffix); +#endif bool empty = strlen(suffix) == 0; delete[] suffix; @@ -3893,9 +3924,9 @@ long long InStream::readLong(long long minv, long long maxv, const std::string & std::vector InStream::readLongs(int size, long long minv, long long maxv, const std::string &variablesName, int indexBase) { - __testlib_readMany(readLongs, readLong(minv, maxv, variablesName), long long, true) if (strict && !variablesName.empty()) validator.addVariable(variablesName); + __testlib_readMany(readLongs, readLong(minv, maxv, variablesName), long long, true) } std::vector InStream::readLongs(int size, int indexBase) { @@ -3939,9 +3970,9 @@ InStream::readUnsignedLong(unsigned long long minv, unsigned long long maxv, con std::vector InStream::readUnsignedLongs(int size, unsigned long long minv, unsigned long long maxv, const std::string &variablesName, int indexBase) { - __testlib_readMany(readUnsignedLongs, readUnsignedLong(minv, maxv, variablesName), unsigned long long, true) if (strict && !variablesName.empty()) validator.addVariable(variablesName); + __testlib_readMany(readUnsignedLongs, readUnsignedLong(minv, maxv, variablesName), unsigned long long, true) } std::vector InStream::readUnsignedLongs(int size, int indexBase) { @@ -3993,9 +4024,9 @@ int InStream::readInteger(int minv, int maxv, const std::string &variableName) { } std::vector InStream::readInts(int size, int minv, int maxv, const std::string &variablesName, int indexBase) { - __testlib_readMany(readInts, readInt(minv, maxv, variablesName), int, true) if (strict && !variablesName.empty()) validator.addVariable(variablesName); + __testlib_readMany(readInts, readInt(minv, maxv, variablesName), int, true) } std::vector InStream::readInts(int size, int indexBase) { @@ -4003,9 +4034,9 @@ std::vector InStream::readInts(int size, int indexBase) { } std::vector InStream::readIntegers(int size, int minv, int maxv, const std::string &variablesName, int indexBase) { - __testlib_readMany(readIntegers, readInt(minv, maxv, variablesName), int, true) if (strict && !variablesName.empty()) validator.addVariable(variablesName); + __testlib_readMany(readIntegers, readInt(minv, maxv, variablesName), int, true) } std::vector InStream::readIntegers(int size, int indexBase) { @@ -4059,9 +4090,9 @@ double InStream::readReal(double minv, double maxv, const std::string &variableN std::vector InStream::readReals(int size, double minv, double maxv, const std::string &variablesName, int indexBase) { - __testlib_readMany(readReals, readReal(minv, maxv, variablesName), double, true) if (strict && !variablesName.empty()) validator.addVariable(variablesName); + __testlib_readMany(readReals, readReal(minv, maxv, variablesName), double, true) } std::vector InStream::readReals(int size, int indexBase) { @@ -4074,9 +4105,9 @@ double InStream::readDouble(double minv, double maxv, const std::string &variabl std::vector InStream::readDoubles(int size, double minv, double maxv, const std::string &variablesName, int indexBase) { - __testlib_readMany(readDoubles, readDouble(minv, maxv, variablesName), double, true) if (strict && !variablesName.empty()) validator.addVariable(variablesName); + __testlib_readMany(readDoubles, readDouble(minv, maxv, variablesName), double, true) } std::vector InStream::readDoubles(int size, int indexBase) { @@ -4126,11 +4157,11 @@ double InStream::readStrictReal(double minv, double maxv, std::vector InStream::readStrictReals(int size, double minv, double maxv, int minAfterPointDigitCount, int maxAfterPointDigitCount, const std::string &variablesName, int indexBase) { + if (strict && !variablesName.empty()) + validator.addVariable(variablesName); __testlib_readMany(readStrictReals, readStrictReal(minv, maxv, minAfterPointDigitCount, maxAfterPointDigitCount, variablesName), double, true) - if (strict && !variablesName.empty()) - validator.addVariable(variablesName); } double InStream::readStrictDouble(double minv, double maxv, @@ -4144,11 +4175,11 @@ double InStream::readStrictDouble(double minv, double maxv, std::vector InStream::readStrictDoubles(int size, double minv, double maxv, int minAfterPointDigitCount, int maxAfterPointDigitCount, const std::string &variablesName, int indexBase) { + if (strict && !variablesName.empty()) + validator.addVariable(variablesName); __testlib_readMany(readStrictDoubles, readStrictDouble(minv, maxv, minAfterPointDigitCount, maxAfterPointDigitCount, variablesName), double, true) - if (strict && !variablesName.empty()) - validator.addVariable(variablesName); } bool InStream::eof() { @@ -4322,9 +4353,9 @@ std::string InStream::readString(const pattern &p, const std::string &variableNa std::vector InStream::readStrings(int size, const pattern &p, const std::string &variablesName, int indexBase) { - __testlib_readMany(readStrings, readString(p, variablesName), std::string, false) if (strict && !variablesName.empty()) validator.addVariable(variablesName); + __testlib_readMany(readStrings, readString(p, variablesName), std::string, false) } std::string InStream::readString(const std::string &ptrn, const std::string &variableName) { @@ -4335,9 +4366,9 @@ std::string InStream::readString(const std::string &ptrn, const std::string &var std::vector InStream::readStrings(int size, const std::string &ptrn, const std::string &variablesName, int indexBase) { pattern p(ptrn); - __testlib_readMany(readStrings, readString(p, variablesName), std::string, false) if (strict && !variablesName.empty()) validator.addVariable(variablesName); + __testlib_readMany(readStrings, readString(p, variablesName), std::string, false) } void InStream::readLineTo(std::string &result) { @@ -4366,9 +4397,9 @@ std::string InStream::readLine(const pattern &p, const std::string &variableName std::vector InStream::readLines(int size, const pattern &p, const std::string &variablesName, int indexBase) { - __testlib_readMany(readLines, readString(p, variablesName), std::string, false) if (strict && !variablesName.empty()) validator.addVariable(variablesName); + __testlib_readMany(readLines, readString(p, variablesName), std::string, false) } std::string InStream::readLine(const std::string &ptrn, const std::string &variableName) { @@ -4378,9 +4409,9 @@ std::string InStream::readLine(const std::string &ptrn, const std::string &varia std::vector InStream::readLines(int size, const std::string &ptrn, const std::string &variablesName, int indexBase) { pattern p(ptrn); - __testlib_readMany(readLines, readString(p, variablesName), std::string, false) if (strict && !variablesName.empty()) validator.addVariable(variablesName); + __testlib_readMany(readLines, readString(p, variablesName), std::string, false) } #ifdef __GNUC__ @@ -4435,7 +4466,7 @@ double __testlib_preparePoints(double points_) { NORETURN void __testlib_quitp(double points, const char *message) { __testlib_points = __testlib_preparePoints(points); - std::string stringPoints = removeDoubleTrailingZeroes(format("%.10f", __testlib_points)); + std::string stringPoints = removeDoubleTrailingZeroes(testlib_format_("%.10f", __testlib_points)); std::string quitMessage; if (NULL == message || 0 == strlen(message)) @@ -4448,7 +4479,7 @@ NORETURN void __testlib_quitp(double points, const char *message) { NORETURN void __testlib_quitp(int points, const char *message) { __testlib_points = __testlib_preparePoints(points); - std::string stringPoints = format("%d", points); + std::string stringPoints = testlib_format_("%d", points); std::string quitMessage; if (NULL == message || 0 == strlen(message)) @@ -4764,7 +4795,7 @@ void registerValidation(int argc, char *argv[]) { if (i + 1 < argc) { long long testCase = stringToLongLong(inf, argv[++i]); if (testCase < 1 || testCase >= __TESTLIB_MAX_TEST_CASE) - quit(_fail, format("Argument testCase should be between 1 and %d, but ", __TESTLIB_MAX_TEST_CASE) + quit(_fail, testlib_format_("Argument testCase should be between 1 and %d, but ", __TESTLIB_MAX_TEST_CASE) + toString(testCase) + " found"); validator.setTestCase(int(testCase)); } else @@ -4922,10 +4953,10 @@ static inline void __testlib_ensure(bool cond, const char *msg) { quit(_fail, msg); } -#define ensure(cond) __testlib_ensure(cond, "Condition failed: \"" #cond "\"") -#define STRINGIZE_DETAIL(x) #x -#define STRINGIZE(x) STRINGIZE_DETAIL(x) -#define ensure_ext(cond) __testlib_ensure(cond, "Line " STRINGIZE(__LINE__) ": Condition failed: \"" #cond "\"") +#define ensure(cond) __testlib_ensure((cond), "Condition failed: \"" #cond "\"") +#define STRINGIZE_DETAIL(x) (#x) +#define STRINGIZE(x) STRINGIZE_DETAIL((x)) +#define ensure_ext(cond) __testlib_ensure((cond), "Line " STRINGIZE(__LINE__) ": Condition failed: \"" #cond "\"") #ifdef __GNUC__ __attribute__ ((format (printf, 2, 3))) @@ -5013,7 +5044,7 @@ void srand(unsigned int seed) RAND_THROW_STATEMENT void startTest(int test) { const std::string testFileName = vtos(test); - if (NULL == freopen(testFileName.c_str(), "wt", stdout)) + if (NULL == testlib_freopen_(testFileName.c_str(), "wt", stdout)) __testlib_fail("Unable to write file '" + testFileName + "'"); } @@ -5182,17 +5213,17 @@ std::vector tokenize(const std::string &s, const std::string &separ NORETURN void __testlib_expectedButFound(TResult result, std::string expected, std::string found, const char *prepend) { std::string message; if (strlen(prepend) != 0) - message = format("%s: expected '%s', but found '%s'", + message = testlib_format_("%s: expected '%s', but found '%s'", compress(prepend).c_str(), compress(expected).c_str(), compress(found).c_str()); else - message = format("expected '%s', but found '%s'", + message = testlib_format_("expected '%s', but found '%s'", compress(expected).c_str(), compress(found).c_str()); quit(result, message); } NORETURN void __testlib_expectedButFound(TResult result, double expected, double found, const char *prepend) { - std::string expectedString = removeDoubleTrailingZeroes(format("%.12f", expected)); - std::string foundString = removeDoubleTrailingZeroes(format("%.12f", found)); + std::string expectedString = removeDoubleTrailingZeroes(testlib_format_("%.12f", expected)); + std::string foundString = removeDoubleTrailingZeroes(testlib_format_("%.12f", found)); __testlib_expectedButFound(result, expectedString, foundString, prepend); } @@ -5223,8 +5254,8 @@ __attribute__ ((format (printf, 4, 5))) #endif NORETURN void expectedButFound(TResult result, double expected, double found, const char *prependFormat, ...) { FMT_TO_RESULT(prependFormat, prependFormat, prepend); - std::string expectedString = removeDoubleTrailingZeroes(format("%.12f", expected)); - std::string foundString = removeDoubleTrailingZeroes(format("%.12f", found)); + std::string expectedString = removeDoubleTrailingZeroes(testlib_format_("%.12f", expected)); + std::string foundString = removeDoubleTrailingZeroes(testlib_format_("%.12f", found)); __testlib_expectedButFound(result, expectedString, foundString, prepend.c_str()); } @@ -6006,7 +6037,11 @@ double deserializePoints(std::string s) { return std::numeric_limits::quiet_NaN(); else { double result; - ensuref(sscanf(s.c_str(), "%lf", &result) == 1, "Invalid serialized points"); +#ifdef _MSC_VER + ensuref(sscanf_s(s.c_str(), "%lf", &result) == 1, "Invalid serialized points"); +#else + ensuref(std::sscanf(s.c_str(), "%lf", &result) == 1, "Invalid serialized points"); +#endif return result; } } @@ -6199,7 +6234,7 @@ std::string opt(const std::string &key, const std::string &default_value) { void ensureNoUnusedOpts() { for (const auto &opt: __testlib_opts) { if (!opt.second.used) { - __testlib_fail(format("Opts: unused key '%s'", compress(opt.first).c_str())); + __testlib_fail(testlib_format_("Opts: unused key '%s'", compress(opt.first).c_str())); } } } @@ -6215,6 +6250,50 @@ void TestlibFinalizeGuard::autoEnsureNoUnusedOpts() { } TestlibFinalizeGuard testlibFinalizeGuard; +#endif + +#ifdef __GNUC__ +__attribute__ ((format (printf, 1, 2))) +#endif +std::string testlib_format_(const char *fmt, ...) { + FMT_TO_RESULT(fmt, fmt, result); + return result; +} + +std::string testlib_format_(const std::string fmt, ...) { + FMT_TO_RESULT(fmt, fmt.c_str(), result); + return result; +} +#if (__cplusplus >= 202002L && __has_include()) || __cpp_lib_format +template +std::string format(const char* fmt, Args&&... args) { + size_t size = size_t(std::snprintf(nullptr, 0, fmt, args...) + 1); + std::vector buffer(size); + std::snprintf(buffer.data(), size, fmt, args...); + return std::string(buffer.data()); +} + +template +std::string format(const std::string fmt, Args&&... args) { + size_t size = size_t(std::snprintf(nullptr, 0, fmt.c_str(), args...) + 1); + std::vector buffer(size); + std::snprintf(buffer.data(), size, fmt.c_str(), args...); + return std::string(buffer.data()); +} +#else +#ifdef __GNUC__ +__attribute__ ((format (printf, 1, 2))) #endif +std::string format(const char *fmt, ...) { + FMT_TO_RESULT(fmt, fmt, result); + return result; +} + +std::string format(const std::string fmt, ...) { + FMT_TO_RESULT(fmt, fmt.c_str(), result); + return result; +} +#endif + #endif diff --git a/tests/scripts/compile b/tests/scripts/compile index 6b31f5f..11357b7 100644 --- a/tests/scripts/compile +++ b/tests/scripts/compile @@ -35,7 +35,15 @@ rm -f "$exe_file" EXTRA_ARGS="" if [[ -z "${TESTLIB_COMPILER_OPTIMIZATION_OPT}" ]]; then - OPTIMIZATION="2" + if [[ "$2" == "--check-only" ]]; then + if [[ "$CPP" == "cl.exe" ]]; then + OPTIMIZATION="d" + else + OPTIMIZATION="0" + fi + else + OPTIMIZATION="2" + fi else OPTIMIZATION="${TESTLIB_COMPILER_OPTIMIZATION_OPT}" fi diff --git a/tests/test-008_format/files/test-format-format1.cpp b/tests/test-008_format/files/test-format-format1.cpp new file mode 100644 index 0000000..5f69d16 --- /dev/null +++ b/tests/test-008_format/files/test-format-format1.cpp @@ -0,0 +1,18 @@ +#if (__cplusplus >= 202002L && __has_include()) +# include +#endif + +#include "testlib.h" + +using namespace std; + +int main(int argc, char** argv) { + registerGen(argc, argv, 1); + + println(format("%d", 42)); + println(format("hello, %s!", "hat")); + println(format("%s%d!", "'hat'", 42)); + println(format("%s%d!", "'%s'", 42)); + + ensure(format("%f", 42.5).substr(0, 4) == "42.5"); +} diff --git a/tests/test-008_format/files/test-format-format2.cpp b/tests/test-008_format/files/test-format-format2.cpp new file mode 100644 index 0000000..f79f39e --- /dev/null +++ b/tests/test-008_format/files/test-format-format2.cpp @@ -0,0 +1,18 @@ +#include "testlib.h" + +#if (__cplusplus >= 202002L && __has_include()) +# include +#endif + +using namespace std; + +int main(int argc, char** argv) { + registerGen(argc, argv, 1); + + println(format("%d", 42)); + println(format("hello, %s!", "hat")); + println(format("%s%d!", "'hat'", 42)); + println(format("%s%d!", "'%s'", 42)); + + ensure(format("%f", 42.5).substr(0, 4) == "42.5"); +} diff --git a/tests/test-008_format/files/test-format.cpp b/tests/test-008_format/files/test-format.cpp new file mode 100644 index 0000000..19da939 --- /dev/null +++ b/tests/test-008_format/files/test-format.cpp @@ -0,0 +1,14 @@ +#include "testlib.h" + +using namespace std; + +int main(int argc, char** argv) { + registerGen(argc, argv, 1); + + println(format("%d", 42)); + println(format("hello, %s!", "hat")); + println(format("%s%d!", "'hat'", 42)); + println(format("%s%d!", "'%s'", 42)); + + ensure(format("%f", 42.5).substr(0, 4) == "42.5"); +} diff --git a/tests/test-008_format/refs/test-format-format1/r1/exit_code b/tests/test-008_format/refs/test-format-format1/r1/exit_code new file mode 100644 index 0000000..573541a --- /dev/null +++ b/tests/test-008_format/refs/test-format-format1/r1/exit_code @@ -0,0 +1 @@ +0 diff --git a/tests/test-008_format/refs/test-format-format1/r1/stderr b/tests/test-008_format/refs/test-format-format1/r1/stderr new file mode 100644 index 0000000..e69de29 diff --git a/tests/test-008_format/refs/test-format-format1/r1/stdout b/tests/test-008_format/refs/test-format-format1/r1/stdout new file mode 100644 index 0000000..e0f278f --- /dev/null +++ b/tests/test-008_format/refs/test-format-format1/r1/stdout @@ -0,0 +1,4 @@ +42 +hello, hat! +'hat'42! +'%s'42! diff --git a/tests/test-008_format/refs/test-format-format1/r2/exit_code b/tests/test-008_format/refs/test-format-format1/r2/exit_code new file mode 100644 index 0000000..573541a --- /dev/null +++ b/tests/test-008_format/refs/test-format-format1/r2/exit_code @@ -0,0 +1 @@ +0 diff --git a/tests/test-008_format/refs/test-format-format1/r2/stderr b/tests/test-008_format/refs/test-format-format1/r2/stderr new file mode 100644 index 0000000..e69de29 diff --git a/tests/test-008_format/refs/test-format-format1/r2/stdout b/tests/test-008_format/refs/test-format-format1/r2/stdout new file mode 100644 index 0000000..e0f278f --- /dev/null +++ b/tests/test-008_format/refs/test-format-format1/r2/stdout @@ -0,0 +1,4 @@ +42 +hello, hat! +'hat'42! +'%s'42! diff --git a/tests/test-008_format/refs/test-format-format2/r1/exit_code b/tests/test-008_format/refs/test-format-format2/r1/exit_code new file mode 100644 index 0000000..573541a --- /dev/null +++ b/tests/test-008_format/refs/test-format-format2/r1/exit_code @@ -0,0 +1 @@ +0 diff --git a/tests/test-008_format/refs/test-format-format2/r1/stderr b/tests/test-008_format/refs/test-format-format2/r1/stderr new file mode 100644 index 0000000..e69de29 diff --git a/tests/test-008_format/refs/test-format-format2/r1/stdout b/tests/test-008_format/refs/test-format-format2/r1/stdout new file mode 100644 index 0000000..e0f278f --- /dev/null +++ b/tests/test-008_format/refs/test-format-format2/r1/stdout @@ -0,0 +1,4 @@ +42 +hello, hat! +'hat'42! +'%s'42! diff --git a/tests/test-008_format/refs/test-format-format2/r2/exit_code b/tests/test-008_format/refs/test-format-format2/r2/exit_code new file mode 100644 index 0000000..573541a --- /dev/null +++ b/tests/test-008_format/refs/test-format-format2/r2/exit_code @@ -0,0 +1 @@ +0 diff --git a/tests/test-008_format/refs/test-format-format2/r2/stderr b/tests/test-008_format/refs/test-format-format2/r2/stderr new file mode 100644 index 0000000..e69de29 diff --git a/tests/test-008_format/refs/test-format-format2/r2/stdout b/tests/test-008_format/refs/test-format-format2/r2/stdout new file mode 100644 index 0000000..e0f278f --- /dev/null +++ b/tests/test-008_format/refs/test-format-format2/r2/stdout @@ -0,0 +1,4 @@ +42 +hello, hat! +'hat'42! +'%s'42! diff --git a/tests/test-008_format/refs/test-format/r1/exit_code b/tests/test-008_format/refs/test-format/r1/exit_code new file mode 100644 index 0000000..573541a --- /dev/null +++ b/tests/test-008_format/refs/test-format/r1/exit_code @@ -0,0 +1 @@ +0 diff --git a/tests/test-008_format/refs/test-format/r1/stderr b/tests/test-008_format/refs/test-format/r1/stderr new file mode 100644 index 0000000..e69de29 diff --git a/tests/test-008_format/refs/test-format/r1/stdout b/tests/test-008_format/refs/test-format/r1/stdout new file mode 100644 index 0000000..e0f278f --- /dev/null +++ b/tests/test-008_format/refs/test-format/r1/stdout @@ -0,0 +1,4 @@ +42 +hello, hat! +'hat'42! +'%s'42! diff --git a/tests/test-008_format/refs/test-format/r2/exit_code b/tests/test-008_format/refs/test-format/r2/exit_code new file mode 100644 index 0000000..573541a --- /dev/null +++ b/tests/test-008_format/refs/test-format/r2/exit_code @@ -0,0 +1 @@ +0 diff --git a/tests/test-008_format/refs/test-format/r2/stderr b/tests/test-008_format/refs/test-format/r2/stderr new file mode 100644 index 0000000..e69de29 diff --git a/tests/test-008_format/refs/test-format/r2/stdout b/tests/test-008_format/refs/test-format/r2/stdout new file mode 100644 index 0000000..e0f278f --- /dev/null +++ b/tests/test-008_format/refs/test-format/r2/stdout @@ -0,0 +1,4 @@ +42 +hello, hat! +'hat'42! +'%s'42! diff --git a/tests/test-008_format/run.sh b/tests/test-008_format/run.sh new file mode 100644 index 0000000..a670325 --- /dev/null +++ b/tests/test-008_format/run.sh @@ -0,0 +1,17 @@ +#!/bin/bash +set -eo pipefail + +bash ../scripts/compile files/test-format.cpp +bash ../scripts/test-ref test-format/r1 ./test-format +bash ../scripts/test-ref test-format/r2 "$VALGRIND" ./test-format +rm -f test-format test-format.exe + +bash ../scripts/compile files/test-format-format1.cpp +bash ../scripts/test-ref test-format-format1/r1 ./test-format-format1 +bash ../scripts/test-ref test-format-format1/r2 "$VALGRIND" ./test-format-format1 +rm -f test-format-format1 test-format-format1.exe + +bash ../scripts/compile files/test-format-format2.cpp +bash ../scripts/test-ref test-format-format2/r1 ./test-format-format2 +bash ../scripts/test-ref test-format-format2/r2 "$VALGRIND" ./test-format-format2 +rm -f test-format-format2 test-format-format2.exe