From b91dc77d6fb192b0e846cac6fc766876564b377c Mon Sep 17 00:00:00 2001 From: Benjamin Sergeant Date: Thu, 14 Mar 2019 18:37:38 -0700 Subject: [PATCH] select interrupt cleanup --- CMakeLists.txt | 10 +- ixwebsocket/IXSelectInterrupt.cpp | 46 ++++++++ ixwebsocket/IXSelectInterrupt.h | 28 +++++ ...entFd.cpp => IXSelectInterruptEventFd.cpp} | 103 ++++++++--------- ixwebsocket/IXSelectInterruptEventFd.h | 32 ++++++ ixwebsocket/IXSelectInterruptFactory.cpp | 25 ++++ ixwebsocket/IXSelectInterruptFactory.h | 15 +++ ixwebsocket/IXSelectInterruptPipe.cpp | 108 ++++++++++++++++++ .../{IXEventFd.h => IXSelectInterruptPipe.h} | 28 ++--- ixwebsocket/IXSocket.cpp | 42 +++---- ixwebsocket/IXSocket.h | 7 +- ixwebsocket/IXSocketFactory.cpp | 28 ++++- ixwebsocket/IXSocketFactory.h | 3 + ixwebsocket/IXWebSocketTransport.cpp | 9 +- test/IXSocketTest.cpp | 21 ++-- test/IXWebSocketServerTest.cpp | 42 ++++--- test/run.py | 2 +- test/test_runner.cpp | 4 - ws/ws.cpp | 5 +- ws/ws_connect.cpp | 1 - ws/ws_receive.cpp | 1 - ws/ws_send.cpp | 1 - 22 files changed, 411 insertions(+), 150 deletions(-) create mode 100644 ixwebsocket/IXSelectInterrupt.cpp create mode 100644 ixwebsocket/IXSelectInterrupt.h rename ixwebsocket/{IXEventFd.cpp => IXSelectInterruptEventFd.cpp} (51%) create mode 100644 ixwebsocket/IXSelectInterruptEventFd.h create mode 100644 ixwebsocket/IXSelectInterruptFactory.cpp create mode 100644 ixwebsocket/IXSelectInterruptFactory.h create mode 100644 ixwebsocket/IXSelectInterruptPipe.cpp rename ixwebsocket/{IXEventFd.h => IXSelectInterruptPipe.h} (50%) diff --git a/CMakeLists.txt b/CMakeLists.txt index 1a416bfe..7515f518 100644 --- a/CMakeLists.txt +++ b/CMakeLists.txt @@ -16,7 +16,6 @@ if (NOT WIN32) endif() set( IXWEBSOCKET_SOURCES - ixwebsocket/IXEventFd.cpp ixwebsocket/IXSocket.cpp ixwebsocket/IXSocketServer.cpp ixwebsocket/IXSocketConnect.cpp @@ -33,10 +32,11 @@ set( IXWEBSOCKET_SOURCES ixwebsocket/IXWebSocketHttpHeaders.cpp ixwebsocket/IXHttpClient.cpp ixwebsocket/IXUrlParser.cpp + ixwebsocket/IXSelectInterrupt.cpp + ixwebsocket/IXSelectInterruptFactory.cpp ) set( IXWEBSOCKET_HEADERS - ixwebsocket/IXEventFd.h ixwebsocket/IXSocket.h ixwebsocket/IXSocketServer.h ixwebsocket/IXSocketConnect.h @@ -58,15 +58,21 @@ set( IXWEBSOCKET_HEADERS ixwebsocket/libwshandshake.hpp ixwebsocket/IXHttpClient.h ixwebsocket/IXUrlParser.h + ixwebsocket/IXSelectInterrupt.h + ixwebsocket/IXSelectInterruptFactory.h ) # Platform specific code if (APPLE) list( APPEND IXWEBSOCKET_SOURCES ixwebsocket/apple/IXSetThreadName_apple.cpp) + list( APPEND IXWEBSOCKET_SOURCES ixwebsocket/IXSelectInterruptPipe.cpp) + list( APPEND IXWEBSOCKET_HEADERS ixwebsocket/IXSelectInterruptPipe.h) elseif (WIN32) list( APPEND IXWEBSOCKET_SOURCES ixwebsocket/windows/IXSetThreadName_windows.cpp) else() list( APPEND IXWEBSOCKET_SOURCES ixwebsocket/linux/IXSetThreadName_linux.cpp) + list( APPEND IXWEBSOCKET_SOURCES ixwebsocket/IXSelectInterruptEventFd.cpp) + list( APPEND IXWEBSOCKET_HEADERS ixwebsocket/IXSelectInterruptEventFd.h) endif() if (USE_TLS) diff --git a/ixwebsocket/IXSelectInterrupt.cpp b/ixwebsocket/IXSelectInterrupt.cpp new file mode 100644 index 00000000..8f7fb2e5 --- /dev/null +++ b/ixwebsocket/IXSelectInterrupt.cpp @@ -0,0 +1,46 @@ +/* + * IXSelectInterrupt.cpp + * Author: Benjamin Sergeant + * Copyright (c) 2019 Machine Zone, Inc. All rights reserved. + */ + +#include "IXSelectInterrupt.h" + +namespace ix +{ + SelectInterrupt::SelectInterrupt() + { + ; + } + + SelectInterrupt::~SelectInterrupt() + { + ; + } + + bool SelectInterrupt::init(std::string& /*errorMsg*/) + { + return true; + } + + bool SelectInterrupt::notify(uint64_t /*value*/) + { + return true; + } + + uint64_t SelectInterrupt::read() + { + return 0; + } + + bool SelectInterrupt::clear() + { + return true; + } + + int SelectInterrupt::getFd() + { + return -1; + } +} + diff --git a/ixwebsocket/IXSelectInterrupt.h b/ixwebsocket/IXSelectInterrupt.h new file mode 100644 index 00000000..7c484a8d --- /dev/null +++ b/ixwebsocket/IXSelectInterrupt.h @@ -0,0 +1,28 @@ +/* + * IXSelectInterrupt.h + * Author: Benjamin Sergeant + * Copyright (c) 2019 Machine Zone, Inc. All rights reserved. + */ + +#pragma once + +#include +#include + +namespace ix +{ + class SelectInterrupt { + public: + SelectInterrupt(); + virtual ~SelectInterrupt(); + + virtual bool init(std::string& errorMsg); + + virtual bool notify(uint64_t value); + virtual bool clear(); + virtual uint64_t read(); + virtual int getFd(); + }; +} + + diff --git a/ixwebsocket/IXEventFd.cpp b/ixwebsocket/IXSelectInterruptEventFd.cpp similarity index 51% rename from ixwebsocket/IXEventFd.cpp rename to ixwebsocket/IXSelectInterruptEventFd.cpp index 1284afdd..a2ab663b 100644 --- a/ixwebsocket/IXEventFd.cpp +++ b/ixwebsocket/IXSelectInterruptEventFd.cpp @@ -1,9 +1,13 @@ /* - * IXEventFd.cpp + * IXSelectInterruptEventFd.cpp * Author: Benjamin Sergeant - * Copyright (c) 2018 Machine Zone, Inc. All rights reserved. + * Copyright (c) 2018-2019 Machine Zone, Inc. All rights reserved. */ +// +// On Linux we use eventd to wake up select. +// + // // Linux/Android has a special type of virtual files. select(2) will react // when reading/writing to those files, unlike closing sockets. @@ -22,57 +26,59 @@ #include "IXEventFd.h" -#ifdef __linux__ -# include -#endif +#include #include // for write +#include // for strerror #include +#include +#include namespace ix { - // File descriptor at index 0 in _fildes is the read end of the pipe - // File descriptor at index 1 in _fildes is the write end of the pipe - const int EventFd::kPipeReadIndex = 0; - const int EventFd::kPipeWriteIndex = 1; - - EventFd::EventFd() + SelectInterruptEventFd::SelectInterruptEventFd() { -#ifdef __linux__ - _eventfd = -1; - _eventfd = eventfd(0, 0); - fcntl(_eventfd, F_SETFL, O_NONBLOCK); -#else - _fildes[kPipeReadIndex] = -1; - _fildes[kPipeWriteIndex] = -1; - - pipe(_fildes); - fcntl(_fildes[kPipeReadIndex], F_SETFL, O_NONBLOCK); - fcntl(_fildes[kPipeWriteIndex], F_SETFL, O_NONBLOCK); -#endif + ; } - EventFd::~EventFd() + SelectInterruptEventFd::~SelectInterruptEventFd() { -#ifdef __linux__ ::close(_eventfd); -#else - ::close(_fildes[kPipeReadIndex]); - ::close(_fildes[kPipeWriteIndex]); - _fildes[kPipeReadIndex] = -1; - _fildes[kPipeWriteIndex] = -1; -#endif } - bool EventFd::notify(uint64_t value) + bool SelectInterruptEventFd::init(std::string& errorMsg) { - int fd; + _eventfd = -1; -#if defined(__linux__) - fd = _eventfd; -#else - fd = _fildes[kPipeWriteIndex]; -#endif + _eventfd = eventfd(0, 0); + if (_eventfd < 0) + { + std::stringstream ss; + ss << "SelectInterruptEventFd::init() failed in eventfd()" + << " : " << strerror(errno); + errorMsg = ss.str(); + + _eventfd = -1; + return false; + } + + if (fcntl(_eventfd, F_SETFL, O_NONBLOCK) == -1) + { + std::stringstream ss; + ss << "SelectInterruptEventFd::init() failed in fcntl() call" + << " : " << strerror(errno); + errorMsg = ss.str(); + + _eventfd = -1; + return false; + } + + return true; + } + + bool SelectInterruptEventFd::notify(uint64_t value) + { + int fd = _eventfd; if (fd == -1) return false; @@ -81,23 +87,17 @@ namespace ix } // TODO: return max uint64_t for errors ? - uint64_t EventFd::read() + uint64_t SelectInterruptEventFd::read() { - int fd; + int fd = _eventfd; -#if defined(__linux__) - fd = _eventfd; -#else - fd = _fildes[kPipeReadIndex]; -#endif uint64_t value = 0; ::read(fd, &value, sizeof(value)); return value; } - bool EventFd::clear() + bool SelectInterruptEventFd::clear() { -#if defined(__linux__) if (_eventfd == -1) return false; // 0 is a special value ; select will not wake up @@ -105,17 +105,10 @@ namespace ix // we should write 8 bytes for an uint64_t return write(_eventfd, &value, sizeof(value)) == 8; -#else - return true; -#endif } - int EventFd::getFd() + int SelectInterruptEventFd::getFd() { -#if defined(__linux__) return _eventfd; -#else - return _fildes[kPipeReadIndex]; -#endif } } diff --git a/ixwebsocket/IXSelectInterruptEventFd.h b/ixwebsocket/IXSelectInterruptEventFd.h new file mode 100644 index 00000000..ba821dc7 --- /dev/null +++ b/ixwebsocket/IXSelectInterruptEventFd.h @@ -0,0 +1,32 @@ +/* + * IXSelectInterruptEventFd.h + * Author: Benjamin Sergeant + * Copyright (c) 2018-2019 Machine Zone, Inc. All rights reserved. + */ + +#pragma once + +#include "IXSelectInterrupt.h" + +#include +#include + +namespace ix +{ + class SelectInterruptEventFd : public SelectInterrupt { + public: + SelectInterruptEventFd(); + virtual ~SelectInterruptEventFd(); + + bool init(std::string& errorMsg) final; + + bool notify(uint64_t value) final; + bool clear() final; + uint64_t read() final; + int getFd() final; + + private: + int _eventfd; + }; +} + diff --git a/ixwebsocket/IXSelectInterruptFactory.cpp b/ixwebsocket/IXSelectInterruptFactory.cpp new file mode 100644 index 00000000..6cf7bf44 --- /dev/null +++ b/ixwebsocket/IXSelectInterruptFactory.cpp @@ -0,0 +1,25 @@ +/* + * IXSelectInterruptFactory.cpp + * Author: Benjamin Sergeant + * Copyright (c) 2019 Machine Zone, Inc. All rights reserved. + */ + +#include "IXSelectInterruptFactory.h" + +#if defined(__linux__) +# include +#else +# include +#endif + +namespace ix +{ + std::shared_ptr createSelectInterrupt() + { +#if defined(__linux__) + return std::make_shared(); +#else + return std::make_shared(); +#endif + } +} diff --git a/ixwebsocket/IXSelectInterruptFactory.h b/ixwebsocket/IXSelectInterruptFactory.h new file mode 100644 index 00000000..11c307e0 --- /dev/null +++ b/ixwebsocket/IXSelectInterruptFactory.h @@ -0,0 +1,15 @@ +/* + * IXSelectInterruptFactory.h + * Author: Benjamin Sergeant + * Copyright (c) 2019 Machine Zone, Inc. All rights reserved. + */ + +#pragma once + +#include + +namespace ix +{ + class SelectInterrupt; + std::shared_ptr createSelectInterrupt(); +} diff --git a/ixwebsocket/IXSelectInterruptPipe.cpp b/ixwebsocket/IXSelectInterruptPipe.cpp new file mode 100644 index 00000000..e05a2efa --- /dev/null +++ b/ixwebsocket/IXSelectInterruptPipe.cpp @@ -0,0 +1,108 @@ +/* + * IXEventFd.cpp + * Author: Benjamin Sergeant + * Copyright (c) 2018-2019 Machine Zone, Inc. All rights reserved. + */ + +// +// On macOS we use UNIX pipes to wake up select. +// + +#include "IXSelectInterruptPipe.h" + +#include // for write +#include // for strerror +#include +#include +#include + +namespace ix +{ + // File descriptor at index 0 in _fildes is the read end of the pipe + // File descriptor at index 1 in _fildes is the write end of the pipe + const int SelectInterruptPipe::kPipeReadIndex = 0; + const int SelectInterruptPipe::kPipeWriteIndex = 1; + + SelectInterruptPipe::SelectInterruptPipe() + { + ; + } + + SelectInterruptPipe::~SelectInterruptPipe() + { + ::close(_fildes[kPipeReadIndex]); + ::close(_fildes[kPipeWriteIndex]); + _fildes[kPipeReadIndex] = -1; + _fildes[kPipeWriteIndex] = -1; + } + + bool SelectInterruptPipe::init(std::string& errorMsg) + { + _fildes[kPipeReadIndex] = -1; + _fildes[kPipeWriteIndex] = -1; + + if (pipe(_fildes) < 0) + { + std::stringstream ss; + ss << "SelectInterruptPipe::init() failed in pipe() call" + << " : " << strerror(errno); + errorMsg = ss.str(); + return false; + } + + if (fcntl(_fildes[kPipeReadIndex], F_SETFL, O_NONBLOCK) == -1) + { + std::stringstream ss; + ss << "SelectInterruptPipe::init() failed in fcntl() call" + << " : " << strerror(errno); + errorMsg = ss.str(); + + _fildes[kPipeReadIndex] = -1; + _fildes[kPipeWriteIndex] = -1; + return false; + } + + if (fcntl(_fildes[kPipeWriteIndex], F_SETFL, O_NONBLOCK) == -1) + { + std::stringstream ss; + ss << "SelectInterruptPipe::init() failed in fcntl() call" + << " : " << strerror(errno); + errorMsg = ss.str(); + + _fildes[kPipeReadIndex] = -1; + _fildes[kPipeWriteIndex] = -1; + return false; + } + + return true; + } + + bool SelectInterruptPipe::notify(uint64_t value) + { + int fd = _fildes[kPipeWriteIndex]; + if (fd == -1) return false; + + // we should write 8 bytes for an uint64_t + return write(fd, &value, sizeof(value)) == 8; + } + + // TODO: return max uint64_t for errors ? + uint64_t SelectInterruptPipe::read() + { + int fd = _fildes[kPipeReadIndex]; + + uint64_t value = 0; + ::read(fd, &value, sizeof(value)); + return value; + } + + bool SelectInterruptPipe::clear() + { + return true; + } + + int SelectInterruptPipe::getFd() + { + return _fildes[kPipeReadIndex]; + } +} diff --git a/ixwebsocket/IXEventFd.h b/ixwebsocket/IXSelectInterruptPipe.h similarity index 50% rename from ixwebsocket/IXEventFd.h rename to ixwebsocket/IXSelectInterruptPipe.h index eae2b172..b874754d 100644 --- a/ixwebsocket/IXEventFd.h +++ b/ixwebsocket/IXSelectInterruptPipe.h @@ -1,37 +1,39 @@ /* - * IXEventFd.h + * IXSelectInterruptPipe.h * Author: Benjamin Sergeant - * Copyright (c) 2018 Machine Zone, Inc. All rights reserved. + * Copyright (c) 2018-2019 Machine Zone, Inc. All rights reserved. */ #pragma once +#include "IXSelectInterrupt.h" + #include +#include namespace ix { - class EventFd { + class SelectInterruptPipe : public SelectInterrupt { public: - EventFd(); - virtual ~EventFd(); + SelectInterruptPipe(); + virtual ~SelectInterruptPipe(); - bool notify(uint64_t value); - bool clear(); - uint64_t read(); - int getFd(); + bool init(std::string& errorMsg) final; + + bool notify(uint64_t value) final; + bool clear() final; + uint64_t read() final; + int getFd() final; private: -#if defined(__linux__) - int _eventfd; -#else // Store file descriptors used by the communication pipe. Communication // happens between a control thread and a background thread, which is // blocked on select. int _fildes[2]; -#endif // Used to identify the read/write idx static const int kPipeReadIndex; static const int kPipeWriteIndex; }; } + diff --git a/ixwebsocket/IXSocket.cpp b/ixwebsocket/IXSocket.cpp index cd9155dd..7e383158 100644 --- a/ixwebsocket/IXSocket.cpp +++ b/ixwebsocket/IXSocket.cpp @@ -7,6 +7,8 @@ #include "IXSocket.h" #include "IXSocketConnect.h" #include "IXNetSystem.h" +#include "IXSelectInterrupt.h" +#include "IXSelectInterruptFactory.h" #include #include @@ -28,7 +30,8 @@ namespace ix constexpr size_t Socket::kChunkSize; Socket::Socket(int fd) : - _sockfd(fd) + _sockfd(fd), + _selectInterrupt(createSelectInterrupt()) { ; } @@ -57,11 +60,11 @@ namespace ix FD_ZERO(&rfds); FD_SET(_sockfd, &rfds); - // File descriptor at index 0 in _fildes is the read end of the pipe - int eventfd = _eventfd.getFd(); - if (eventfd != -1) + // File descriptor used to interrupt select when needed + int interruptFd = _selectInterrupt->getFd(); + if (interruptFd != -1) { - FD_SET(eventfd, &rfds); + FD_SET(interruptFd, &rfds); } struct timeval timeout; @@ -70,7 +73,7 @@ namespace ix // Compute the highest fd. int sockfd = _sockfd; - int nfds = (std::max)(sockfd, eventfd); + int nfds = (std::max)(sockfd, interruptFd); int ret = ::select(nfds + 1, &rfds, nullptr, nullptr, (timeoutSecs < 0) ? nullptr : &timeout); @@ -84,9 +87,9 @@ namespace ix { pollResult = PollResultType_Timeout; } - else if (eventfd != -1 && FD_ISSET(eventfd, &rfds)) + else if (interruptFd != -1 && FD_ISSET(interruptFd, &rfds)) { - uint64_t value = _eventfd.read(); + uint64_t value = _selectInterrupt->read(); if (value == kSendRequest) { @@ -104,7 +107,7 @@ namespace ix // Wake up from poll/select by writing to the pipe which is watched by select bool Socket::wakeUpFromPoll(uint8_t wakeUpCode) { - return _eventfd.notify(wakeUpCode); + return _selectInterrupt->notify(wakeUpCode); } bool Socket::connect(const std::string& host, @@ -114,7 +117,7 @@ namespace ix { std::lock_guard lock(_socketMutex); - if (!_eventfd.clear()) return false; + if (!_selectInterrupt->clear()) return false; _sockfd = SocketConnect::connect(host, port, errMsg, isCancellationRequested); return _sockfd != -1; @@ -173,24 +176,9 @@ namespace ix #endif } - bool Socket::init() + bool Socket::init(std::string& errorMsg) { -#ifdef _WIN32 - INT rc; - WSADATA wsaData; - - rc = WSAStartup(MAKEWORD(2, 2), &wsaData); - return rc != 0; -#else - return true; -#endif - } - - void Socket::cleanup() - { -#ifdef _WIN32 - WSACleanup(); -#endif + return _selectInterrupt->init(errorMsg); } bool Socket::writeBytes(const std::string& str, diff --git a/ixwebsocket/IXSocket.h b/ixwebsocket/IXSocket.h index c336390e..ad9a09b1 100644 --- a/ixwebsocket/IXSocket.h +++ b/ixwebsocket/IXSocket.h @@ -23,6 +23,8 @@ typedef SSIZE_T ssize_t; namespace ix { + class SelectInterrupt; + enum PollResultType { PollResultType_ReadyForRead = 0, @@ -38,6 +40,7 @@ namespace ix Socket(int fd = -1); virtual ~Socket(); + bool init(std::string& errorMsg); void configure(); @@ -72,8 +75,6 @@ namespace ix const CancellationRequest& isCancellationRequested); static int getErrno(); - static bool init(); // Required on Windows to initialize WinSocket - static void cleanup(); // Required on Windows to cleanup WinSocket // Used as special codes for pipe communication static const uint64_t kSendRequest; @@ -93,6 +94,6 @@ namespace ix std::vector _readBuffer; static constexpr size_t kChunkSize = 1 << 15; - EventFd _eventfd; + std::shared_ptr _selectInterrupt; }; } diff --git a/ixwebsocket/IXSocketFactory.cpp b/ixwebsocket/IXSocketFactory.cpp index 22d32a1b..2c574757 100644 --- a/ixwebsocket/IXSocketFactory.cpp +++ b/ixwebsocket/IXSocketFactory.cpp @@ -20,23 +20,45 @@ namespace ix std::string& errorMsg) { errorMsg.clear(); + std::shared_ptr socket; if (!tls) { - return std::make_shared(); + socket = std::make_shared(); } else { #ifdef IXWEBSOCKET_USE_TLS # ifdef __APPLE__ - return std::make_shared(); + socket = std::make_shared(); # else - return std::make_shared(); + socket = std::make_shared(); # endif #else errorMsg = "TLS support is not enabled on this platform."; return nullptr; #endif } + + if (!socket->init(errorMsg)) + { + socket.reset(); + } + + return socket; + } + + std::shared_ptr createSocket(int fd, + std::string& errorMsg) + { + errorMsg.clear(); + + std::shared_ptr socket = std::make_shared(fd); + if (!socket->init(errorMsg)) + { + socket.reset(); + } + + return socket; } } diff --git a/ixwebsocket/IXSocketFactory.h b/ixwebsocket/IXSocketFactory.h index 518e0c29..c10145d1 100644 --- a/ixwebsocket/IXSocketFactory.h +++ b/ixwebsocket/IXSocketFactory.h @@ -14,4 +14,7 @@ namespace ix class Socket; std::shared_ptr createSocket(bool tls, std::string& errorMsg); + + std::shared_ptr createSocket(int fd, + std::string& errorMsg); } diff --git a/ixwebsocket/IXWebSocketTransport.cpp b/ixwebsocket/IXWebSocketTransport.cpp index ea33c2ad..1158d260 100644 --- a/ixwebsocket/IXWebSocketTransport.cpp +++ b/ixwebsocket/IXWebSocketTransport.cpp @@ -123,8 +123,13 @@ namespace ix // Server WebSocketInitResult WebSocketTransport::connectToSocket(int fd, int timeoutSecs) { - _socket.reset(); - _socket = std::make_shared(fd); + std::string errorMsg; + _socket = createSocket(fd, errorMsg); + + if (!_socket) + { + return WebSocketInitResult(false, 0, errorMsg); + } WebSocketHandshake webSocketHandshake(_requestInitCancellation, _socket, diff --git a/test/IXSocketTest.cpp b/test/IXSocketTest.cpp index 0c409705..0703f1f5 100644 --- a/test/IXSocketTest.cpp +++ b/test/IXSocketTest.cpp @@ -5,17 +5,10 @@ */ #include +#include #include #include -#if defined(__APPLE__) or defined(__linux__) -# ifdef __APPLE__ -# include -# else -# include -# endif -#endif - #include "IXTest.h" #include "catch.hpp" #include @@ -63,7 +56,9 @@ TEST_CASE("socket", "[socket]") { SECTION("Connect to google HTTP server. Send GET request without header. Should return 200") { - std::shared_ptr socket(new Socket); + std::string errMsg; + bool tls = false; + std::shared_ptr socket = createSocket(tls, errMsg); std::string host("www.google.com"); int port = 80; @@ -82,11 +77,9 @@ TEST_CASE("socket", "[socket]") #if defined(__APPLE__) or defined(__linux__) SECTION("Connect to google HTTPS server. Send GET request without header. Should return 200") { -# ifdef __APPLE__ - std::shared_ptr socket = std::make_shared(); -# else - std::shared_ptr socket = std::make_shared(); -# endif + std::string errMsg; + bool tls = true; + std::shared_ptr socket = createSocket(tls, errMsg); std::string host("www.google.com"); int port = 443; std::string request("GET / HTTP/1.1\r\n\r\n"); diff --git a/test/IXWebSocketServerTest.cpp b/test/IXWebSocketServerTest.cpp index 7a3c1607..ab7d49e4 100644 --- a/test/IXWebSocketServerTest.cpp +++ b/test/IXWebSocketServerTest.cpp @@ -8,6 +8,7 @@ #include #include #include +#include #include "IXTest.h" @@ -79,17 +80,18 @@ TEST_CASE("Websocket_server", "[websocket_server]") ix::WebSocketServer server(port); REQUIRE(startServer(server)); - Socket socket; - std::string host("localhost"); std::string errMsg; + bool tls = false; + std::shared_ptr socket = createSocket(tls, errMsg); + std::string host("localhost"); auto isCancellationRequested = []() -> bool { return false; }; - bool success = socket.connect(host, port, errMsg, isCancellationRequested); + bool success = socket->connect(host, port, errMsg, isCancellationRequested); REQUIRE(success); - auto lineResult = socket.readLine(isCancellationRequested); + auto lineResult = socket->readLine(isCancellationRequested); auto lineValid = lineResult.first; auto line = lineResult.second; @@ -111,20 +113,21 @@ TEST_CASE("Websocket_server", "[websocket_server]") ix::WebSocketServer server(port); REQUIRE(startServer(server)); - Socket socket; - std::string host("localhost"); std::string errMsg; + bool tls = false; + std::shared_ptr socket = createSocket(tls, errMsg); + std::string host("localhost"); auto isCancellationRequested = []() -> bool { return false; }; - bool success = socket.connect(host, port, errMsg, isCancellationRequested); + bool success = socket->connect(host, port, errMsg, isCancellationRequested); REQUIRE(success); Logger() << "writeBytes"; - socket.writeBytes("GET /\r\n", isCancellationRequested); + socket->writeBytes("GET /\r\n", isCancellationRequested); - auto lineResult = socket.readLine(isCancellationRequested); + auto lineResult = socket->readLine(isCancellationRequested); auto lineValid = lineResult.first; auto line = lineResult.second; @@ -146,24 +149,25 @@ TEST_CASE("Websocket_server", "[websocket_server]") ix::WebSocketServer server(port); REQUIRE(startServer(server)); - Socket socket; - std::string host("localhost"); std::string errMsg; + bool tls = false; + std::shared_ptr socket = createSocket(tls, errMsg); + std::string host("localhost"); auto isCancellationRequested = []() -> bool { return false; }; - bool success = socket.connect(host, port, errMsg, isCancellationRequested); + bool success = socket->connect(host, port, errMsg, isCancellationRequested); REQUIRE(success); - socket.writeBytes("GET / HTTP/1.1\r\n" - "Upgrade: websocket\r\n" - "Sec-WebSocket-Version: 13\r\n" - "Sec-WebSocket-Key: foobar\r\n" - "\r\n", - isCancellationRequested); + socket->writeBytes("GET / HTTP/1.1\r\n" + "Upgrade: websocket\r\n" + "Sec-WebSocket-Version: 13\r\n" + "Sec-WebSocket-Key: foobar\r\n" + "\r\n", + isCancellationRequested); - auto lineResult = socket.readLine(isCancellationRequested); + auto lineResult = socket->readLine(isCancellationRequested); auto lineValid = lineResult.first; auto line = lineResult.second; diff --git a/test/run.py b/test/run.py index 955deb9f..e75327dd 100644 --- a/test/run.py +++ b/test/run.py @@ -78,7 +78,7 @@ shutil.copy(os.path.join( 'bin', 'zlib.dll'), '.') -lldb = "lldb --batch -o 'run' -k 'thread backtrace all' -k 'quit 1'" +# lldb = "lldb --batch -o 'run' -k 'thread backtrace all' -k 'quit 1'" lldb = "" # Disabled for now testCommand = '{} {} {}'.format(lldb, testBinary, os.getenv('TEST', '')) ret = os.system(testCommand) diff --git a/test/test_runner.cpp b/test/test_runner.cpp index 2eb6ee1c..77d71dfa 100644 --- a/test/test_runner.cpp +++ b/test/test_runner.cpp @@ -11,10 +11,6 @@ int main(int argc, char* argv[]) { - ix::Socket::init(); // for Windows - int result = Catch::Session().run(argc, argv); - - ix::Socket::cleanup(); // for Windows return result; } diff --git a/ws/ws.cpp b/ws/ws.cpp index dfaa6fc9..eff1c115 100644 --- a/ws/ws.cpp +++ b/ws/ws.cpp @@ -92,12 +92,9 @@ int main(int argc, char** argv) CLI11_PARSE(app, argc, argv); - ix::Socket::init(); - - // pid file handling - if (app.got_subcommand("transfer")) { + // pid file handling if (!pidfile.empty()) { unlink(pidfile.c_str()); diff --git a/ws/ws_connect.cpp b/ws/ws_connect.cpp index 9fda794d..2889bead 100644 --- a/ws/ws_connect.cpp +++ b/ws/ws_connect.cpp @@ -153,7 +153,6 @@ namespace ix int ws_connect_main(const std::string& url) { - Socket::init(); interactiveMain(url); return 0; } diff --git a/ws/ws_receive.cpp b/ws/ws_receive.cpp index 0bfaf1c3..15d38ee0 100644 --- a/ws/ws_receive.cpp +++ b/ws/ws_receive.cpp @@ -254,7 +254,6 @@ namespace ix int ws_receive_main(const std::string& url, bool enablePerMessageDeflate) { - Socket::init(); wsReceive(url, enablePerMessageDeflate); return 0; } diff --git a/ws/ws_send.cpp b/ws/ws_send.cpp index 2c61478b..353a55f9 100644 --- a/ws/ws_send.cpp +++ b/ws/ws_send.cpp @@ -298,7 +298,6 @@ namespace ix bool throttle = false; bool enablePerMessageDeflate = false; - Socket::init(); wsSend(url, path, enablePerMessageDeflate, throttle); return 0; }