This commit is contained in:
David Matson 2025-03-26 14:14:20 -07:00 committed by GitHub
commit 3b0e68167b
No known key found for this signature in database
GPG key ID: B5690EEEBB952194
6 changed files with 152 additions and 27 deletions

View file

@ -14,6 +14,7 @@ Chris Taylor <taylorc@google.com>
Dan Egnor <egnor@google.com>
Dave MacLachlan <dmaclach@gmail.com>
David Anderson <danderson@google.com>
David Matson
Dean Sturtevant
Eric Roman <eroman@chromium.org>
Gene Volovich <gv@cite.com>

View file

@ -701,7 +701,8 @@ typedef struct _RTL_CRITICAL_SECTION GTEST_CRITICAL_SECTION;
#if defined(GTEST_OS_LINUX) || defined(GTEST_OS_GNU_KFREEBSD) || \
defined(GTEST_OS_DRAGONFLY) || defined(GTEST_OS_FREEBSD) || \
defined(GTEST_OS_NETBSD) || defined(GTEST_OS_OPENBSD) || \
defined(GTEST_OS_GNU_HURD) || defined(GTEST_OS_MAC)
defined(GTEST_OS_GNU_HURD) || defined(GTEST_OS_MAC) || \
defined(GTEST_OS_WINDOWS)
#define GTEST_CAN_STREAM_RESULTS_ 1
#else
#define GTEST_CAN_STREAM_RESULTS_ 0
@ -909,6 +910,18 @@ typedef struct _RTL_CRITICAL_SECTION GTEST_CRITICAL_SECTION;
#define GTEST_ATTRIBUTE_NO_SANITIZE_THREAD_
#endif
#if GTEST_CAN_STREAM_RESULTS_
#if GTEST_OS_WINDOWS
#define NOMINMAX
#include <WinSock2.h> // NOLINT
#else // GTEST_OS_WINDOWS
#include <arpa/inet.h> // NOLINT
#include <netdb.h> // NOLINT
#endif // GTEST_OS_WINDOWS
#endif // GTEST_CAN_STREAM_RESULTS_
namespace testing {
class Message;
@ -2159,6 +2172,27 @@ GTEST_DISABLE_MSC_DEPRECATED_POP_()
[[noreturn]] inline void Abort() { abort(); }
#endif // GTEST_OS_WINDOWS_MOBILE
// Sockets porting.
#if GTEST_CAN_STREAM_RESULTS_
int SocketStartup();
int Socket(int domain, int type, int protocol);
int GetAddrInfo(const char* nodename,
const char* servname,
const struct addrinfo* hints,
struct addrinfo** res);
void FreeAddrInfo(struct addrinfo* ai);
int Connect(int sockfd, const struct sockaddr* addr, size_t addrlen);
const char* GaiStrError(int errcode);
#endif // GTEST_CAN_STREAM_RESULTS_
} // namespace posix
// MSVC "deprecates" snprintf and issues warnings wherever it is used. In

View file

@ -51,11 +51,6 @@
#include "gtest/internal/gtest-port.h"
#if GTEST_CAN_STREAM_RESULTS_
#include <arpa/inet.h> // NOLINT
#include <netdb.h> // NOLINT
#endif
#ifdef GTEST_OS_WINDOWS
#include <windows.h> // NOLINT
#endif // GTEST_OS_WINDOWS
@ -1075,7 +1070,7 @@ class TestResultAccessor {
#if GTEST_CAN_STREAM_RESULTS_
// Streams test results to the given port on the given host machine.
class StreamingListener : public EmptyTestEventListener {
class GTEST_API_ StreamingListener : public EmptyTestEventListener {
public:
// Abstract base class for writing strings to a socket.
class AbstractSocketWriter {
@ -1109,8 +1104,9 @@ class StreamingListener : public EmptyTestEventListener {
GTEST_CHECK_(sockfd_ != -1)
<< "Send() can be called only when there is a connection.";
const auto len = static_cast<size_t>(message.length());
if (write(sockfd_, message.c_str(), len) != static_cast<ssize_t>(len)) {
const auto len = static_cast<unsigned int>(message.length());
if (posix::Write(sockfd_, message.c_str(), len) !=
static_cast<int>(len)) {
GTEST_LOG_(WARNING) << "stream_result_to: failed to stream to "
<< host_name_ << ":" << port_num_;
}
@ -1125,7 +1121,7 @@ class StreamingListener : public EmptyTestEventListener {
GTEST_CHECK_(sockfd_ != -1)
<< "CloseConnection() can be called only when there is a connection.";
close(sockfd_);
posix::Close(sockfd_);
sockfd_ = -1;
}

View file

@ -1430,5 +1430,98 @@ const char* StringFromGTestEnv(const char* flag, const char* default_value) {
#endif // defined(GTEST_GET_STRING_FROM_ENV_)
}
#if GTEST_CAN_STREAM_RESULTS_
namespace posix {
#if GTEST_OS_WINDOWS
#include <WinSock2.h> // NOLINT
#include <ws2tcpip.h> // NOLINT
#pragma comment(lib, "Ws2_32.lib")
int SocketStartup() {
WSADATA winsockVersion{};
int startupError{ WSAStartup(MAKEWORD(2, 2), &winsockVersion) };
if (startupError)
{
return startupError;
}
if (LOBYTE(winsockVersion.wVersion) != 2 ||
HIBYTE(winsockVersion.wVersion) != 2)
{
return -1;
}
return 0;
}
int Socket(int domain, int type, int protocol)
{
return static_cast<int>(socket(domain, type, protocol));
}
int GetAddrInfo(const char* nodename,
const char* servname,
const struct addrinfo* hints,
struct addrinfo** res)
{
return getaddrinfo(nodename, servname, hints, res);
}
void FreeAddrInfo(struct addrinfo* ai)
{
freeaddrinfo(ai);
}
int Connect(int sockfd, const struct sockaddr* addr, size_t addrlen)
{
return connect(sockfd, addr, static_cast<int>(addrlen));
}
const char* GaiStrError(int errcode)
{
return gai_strerrorA(errcode);
}
#else // GTEST_OS_WINDOWS
#include <arpa/inet.h> // NOLINT
#include <netdb.h> // NOLINT
#include <sys/socket.h> // NOLINT
#include <sys/types.h> // NOLINT
int SocketStartup() {
return 0;
}
int Socket(int domain, int type, int protocol) {
return socket(domain, type, protocol);
}
int GetAddrInfo(const char* nodename,
const char* servname,
const struct addrinfo* hints,
struct addrinfo** res) {
return getaddrinfo(nodename, servname, hints, res);
}
void FreeAddrInfo(struct addrinfo* ai) { freeaddrinfo(ai); }
int Connect(int sockfd, const struct sockaddr* addr, size_t addrlen) {
return connect(sockfd, addr, static_cast<socklen_t>(addrlen));
}
const char* GaiStrError(int errcode) { return gai_strerror(errcode); }
#endif // GTEST_OS_WINDOWS
} // namespace posix
#endif // GTEST_CAN_STREAM_RESULTS_
} // namespace internal
} // namespace testing

View file

@ -122,13 +122,6 @@
#include <stdexcept>
#endif
#if GTEST_CAN_STREAM_RESULTS_
#include <arpa/inet.h> // NOLINT
#include <netdb.h> // NOLINT
#include <sys/socket.h> // NOLINT
#include <sys/types.h> // NOLINT
#endif
#include "src/gtest-internal-inl.h"
#ifdef GTEST_OS_WINDOWS
@ -390,7 +383,7 @@ GTEST_DEFINE_string_(
testing::internal::StringFromGTestEnv("stream_result_to", ""),
"This flag specifies the host name and the port number on which to stream "
"test results. Example: \"localhost:555\". The flag is effective only on "
"Linux and macOS.");
"Linux, macOS, and Windows.");
GTEST_DEFINE_bool_(
throw_on_failure,
@ -5017,6 +5010,11 @@ void StreamingListener::SocketWriter::MakeConnection() {
GTEST_CHECK_(sockfd_ == -1)
<< "MakeConnection() can't be called when there is already a connection.";
if (!posix::SocketStartup()) {
GTEST_LOG_(WARNING) << "Socket startup failed.";
return;
}
addrinfo hints;
memset(&hints, 0, sizeof(hints));
hints.ai_family = AF_UNSPEC; // To allow both IPv4 and IPv6 addresses.
@ -5025,28 +5023,31 @@ void StreamingListener::SocketWriter::MakeConnection() {
// Use the getaddrinfo() to get a linked list of IP addresses for
// the given host name.
const int error_num =
getaddrinfo(host_name_.c_str(), port_num_.c_str(), &hints, &servinfo);
const int error_num = posix::GetAddrInfo(host_name_.c_str(),
port_num_.c_str(),
&hints,
&servinfo);
if (error_num != 0) {
GTEST_LOG_(WARNING) << "stream_result_to: getaddrinfo() failed: "
<< gai_strerror(error_num);
<< posix::GaiStrError(error_num);
}
// Loop through all the results and connect to the first we can.
for (addrinfo* cur_addr = servinfo; sockfd_ == -1 && cur_addr != nullptr;
cur_addr = cur_addr->ai_next) {
sockfd_ = socket(cur_addr->ai_family, cur_addr->ai_socktype,
cur_addr->ai_protocol);
sockfd_ = posix::Socket(cur_addr->ai_family, cur_addr->ai_socktype,
cur_addr->ai_protocol);
if (sockfd_ != -1) {
// Connect the client socket to the server socket.
if (connect(sockfd_, cur_addr->ai_addr, cur_addr->ai_addrlen) == -1) {
close(sockfd_);
if (posix::Connect(sockfd_, cur_addr->ai_addr, cur_addr->ai_addrlen) ==
-1) {
posix::Close(sockfd_);
sockfd_ = -1;
}
}
}
freeaddrinfo(servinfo); // all done with this structure
posix::FreeAddrInfo(servinfo); // all done with this structure
if (sockfd_ == -1) {
GTEST_LOG_(WARNING) << "stream_result_to: failed to connect to "

View file

@ -144,7 +144,7 @@ class GTestHelpTest(gtest_test_utils.TestCase):
self.assertTrue(HELP_REGEX.search(output), output)
if IS_DARWIN or IS_LINUX or IS_GNUHURD or is_bsd_based_os():
if IS_DARWIN or IS_LINUX or IS_GNUHURD or is_bsd_based_os() or IS_WINDOWS:
self.assertIn(STREAM_RESULT_TO_FLAG, output)
else:
self.assertNotIn(STREAM_RESULT_TO_FLAG, output)