fmt/test/gtest-extra.cc

88 lines
2.2 KiB
C++
Raw Normal View History

2018-03-04 20:16:51 +03:00
// Formatting library for C++ - custom Google Test assertions
//
// Copyright (c) 2012 - present, Victor Zverovich
// All rights reserved.
//
// For the license information refer to format.h.
#include "gtest-extra.h"
2019-11-15 16:41:14 +03:00
#if FMT_USE_FCNTL
2018-05-19 20:32:53 +03:00
using fmt::file;
void OutputRedirect::flush() {
2019-01-13 05:27:38 +03:00
# if EOF != -1
# error "FMT_RETRY assumes return value of -1 indicating failure"
# endif
2014-05-04 20:01:18 +04:00
int result = 0;
FMT_RETRY(result, fflush(file_));
2019-01-13 05:27:38 +03:00
if (result != 0) throw fmt::system_error(errno, "cannot flush stream");
2014-05-04 20:01:18 +04:00
}
void OutputRedirect::restore() {
2019-01-13 05:27:38 +03:00
if (original_.descriptor() == -1) return; // Already restored.
flush();
2014-05-04 20:01:18 +04:00
// Restore the original file.
original_.dup2(FMT_POSIX(fileno(file_)));
2014-05-04 21:08:29 +04:00
original_.close();
2014-05-04 20:01:18 +04:00
}
2019-01-13 05:27:38 +03:00
OutputRedirect::OutputRedirect(FILE* f) : file_(f) {
flush();
2018-05-19 20:32:53 +03:00
int fd = FMT_POSIX(fileno(f));
// Create a file object referring to the original file.
original_ = file::dup(fd);
2014-05-04 03:47:00 +04:00
// Create a pipe.
2018-05-19 20:32:53 +03:00
file write_end;
file::pipe(read_end_, write_end);
2014-05-04 20:01:18 +04:00
// Connect the passed FILE object to the write end of the pipe.
2014-05-03 23:28:02 +04:00
write_end.dup2(fd);
}
OutputRedirect::~OutputRedirect() FMT_NOEXCEPT {
2014-05-04 03:47:00 +04:00
try {
restore();
2019-01-13 05:27:38 +03:00
} catch (const std::exception& e) {
2014-05-04 21:08:29 +04:00
std::fputs(e.what(), stderr);
2014-05-04 03:47:00 +04:00
}
}
std::string OutputRedirect::restore_and_read() {
// Restore output.
restore();
2014-05-03 22:26:46 +04:00
// Read everything from the pipe.
std::string content;
2019-01-13 05:27:38 +03:00
if (read_end_.descriptor() == -1) return content; // Already read.
2014-05-03 22:26:46 +04:00
enum { BUFFER_SIZE = 4096 };
char buffer[BUFFER_SIZE];
2016-03-02 18:53:14 +03:00
std::size_t count = 0;
2014-05-03 22:26:46 +04:00
do {
2014-05-03 23:28:02 +04:00
count = read_end_.read(buffer, BUFFER_SIZE);
2016-03-02 18:53:14 +03:00
content.append(buffer, count);
2014-05-03 22:26:46 +04:00
} while (count != 0);
2014-05-04 21:08:29 +04:00
read_end_.close();
2014-05-03 22:26:46 +04:00
return content;
}
2019-01-13 05:27:38 +03:00
std::string read(file& f, std::size_t count) {
std::string buffer(count, '\0');
2016-03-02 18:53:14 +03:00
std::size_t n = 0, offset = 0;
do {
n = f.read(&buffer[offset], count - offset);
// We can't read more than size_t bytes since count has type size_t.
offset += n;
} while (offset < count && n != 0);
buffer.resize(offset);
return buffer;
}
#endif // FMT_USE_FILE_DESCRIPTORS
2014-05-06 17:11:39 +04:00
std::string format_system_error(int error_code, fmt::string_view message) {
fmt::memory_buffer out;
format_system_error(out, error_code, message);
return to_string(out);
2014-05-06 17:11:39 +04:00
}