| // Copyright (c) 2014 The Chromium Authors. All rights reserved. | 
 | // Use of this source code is governed by a BSD-style license that can be | 
 | // found in the LICENSE file. | 
 |  | 
 | // This unit test relies on /proc, which is not available on non-Linux based | 
 | // OSes that we support. | 
 | #if defined(__linux__) | 
 |  | 
 | #include "quiche/quic/tools/quic_default_client.h" | 
 |  | 
 | #include <dirent.h> | 
 | #include <sys/types.h> | 
 |  | 
 | #include <memory> | 
 | #include <string> | 
 | #include <utility> | 
 |  | 
 | #include "absl/strings/match.h" | 
 | #include "absl/strings/string_view.h" | 
 | #include "quiche/quic/core/io/quic_default_event_loop.h" | 
 | #include "quiche/quic/core/io/quic_event_loop.h" | 
 | #include "quiche/quic/core/quic_default_clock.h" | 
 | #include "quiche/quic/platform/api/quic_test.h" | 
 | #include "quiche/quic/platform/api/quic_test_loopback.h" | 
 | #include "quiche/quic/test_tools/crypto_test_utils.h" | 
 | #include "quiche/common/quiche_text_utils.h" | 
 |  | 
 | namespace quic { | 
 | namespace test { | 
 | namespace { | 
 |  | 
 | const char* kPathToFds = "/proc/self/fd"; | 
 |  | 
 | // Return the value of a symbolic link in |path|, if |path| is not found, return | 
 | // an empty string. | 
 | std::string ReadLink(const std::string& path) { | 
 |   std::string result(PATH_MAX, '\0'); | 
 |   ssize_t result_size = readlink(path.c_str(), &result[0], result.size()); | 
 |   if (result_size < 0 && errno == ENOENT) { | 
 |     return ""; | 
 |   } | 
 |   QUICHE_CHECK(result_size > 0 && | 
 |                static_cast<size_t>(result_size) < result.size()) | 
 |       << "result_size:" << result_size << ", errno:" << errno | 
 |       << ", path:" << path; | 
 |   result.resize(result_size); | 
 |   return result; | 
 | } | 
 |  | 
 | // Counts the number of open sockets for the current process. | 
 | size_t NumOpenSocketFDs() { | 
 |   size_t socket_count = 0; | 
 |   dirent* file; | 
 |   std::unique_ptr<DIR, int (*)(DIR*)> fd_directory(opendir(kPathToFds), | 
 |                                                    closedir); | 
 |   while ((file = readdir(fd_directory.get())) != nullptr) { | 
 |     absl::string_view name(file->d_name); | 
 |     if (name == "." || name == "..") { | 
 |       continue; | 
 |     } | 
 |  | 
 |     std::string fd_path = ReadLink(absl::StrCat(kPathToFds, "/", name)); | 
 |     if (absl::StartsWith(fd_path, "socket:")) { | 
 |       socket_count++; | 
 |     } | 
 |   } | 
 |   return socket_count; | 
 | } | 
 |  | 
 | class QuicDefaultClientTest : public QuicTest { | 
 |  public: | 
 |   QuicDefaultClientTest() | 
 |       : event_loop_(GetDefaultEventLoop()->Create(QuicDefaultClock::Get())) { | 
 |     // Creates and destroys a single client first which may open persistent | 
 |     // sockets when initializing platform dependencies like certificate | 
 |     // verifier. Future creation of addtional clients will deterministically | 
 |     // open one socket per client. | 
 |     CreateAndInitializeQuicClient(); | 
 |   } | 
 |  | 
 |   // Creates a new QuicClient and Initializes it on an unused port. | 
 |   // Caller is responsible for deletion. | 
 |   std::unique_ptr<QuicDefaultClient> CreateAndInitializeQuicClient() { | 
 |     QuicSocketAddress server_address(QuicSocketAddress(TestLoopback(), 0)); | 
 |     QuicServerId server_id("hostname", server_address.port()); | 
 |     ParsedQuicVersionVector versions = AllSupportedVersions(); | 
 |     auto client = std::make_unique<QuicDefaultClient>( | 
 |         server_address, server_id, versions, event_loop_.get(), | 
 |         crypto_test_utils::ProofVerifierForTesting()); | 
 |     EXPECT_TRUE(client->Initialize()); | 
 |     return client; | 
 |   } | 
 |  | 
 |  private: | 
 |   std::unique_ptr<QuicEventLoop> event_loop_; | 
 | }; | 
 |  | 
 | TEST_F(QuicDefaultClientTest, DoNotLeakSocketFDs) { | 
 |   // Make sure that the QuicClient doesn't leak socket FDs. Doing so could cause | 
 |   // port exhaustion in long running processes which repeatedly create clients. | 
 |  | 
 |   // Record the initial number of FDs. | 
 |   size_t number_of_open_fds = NumOpenSocketFDs(); | 
 |  | 
 |   // Create a number of clients, initialize them, and verify this has resulted | 
 |   // in additional FDs being opened. | 
 |   const int kNumClients = 50; | 
 |   for (int i = 0; i < kNumClients; ++i) { | 
 |     EXPECT_EQ(number_of_open_fds, NumOpenSocketFDs()); | 
 |     std::unique_ptr<QuicDefaultClient> client(CreateAndInitializeQuicClient()); | 
 |     // Initializing the client will create a new FD. | 
 |     EXPECT_EQ(number_of_open_fds + 1, NumOpenSocketFDs()); | 
 |   } | 
 |  | 
 |   // The FDs created by the QuicClients should now be closed. | 
 |   EXPECT_EQ(number_of_open_fds, NumOpenSocketFDs()); | 
 | } | 
 |  | 
 | TEST_F(QuicDefaultClientTest, CreateAndCleanUpUDPSockets) { | 
 |   size_t number_of_open_fds = NumOpenSocketFDs(); | 
 |  | 
 |   std::unique_ptr<QuicDefaultClient> client(CreateAndInitializeQuicClient()); | 
 |   // Creating and initializing a client will result in one socket being opened. | 
 |   EXPECT_EQ(number_of_open_fds + 1, NumOpenSocketFDs()); | 
 |  | 
 |   // Create more UDP sockets. | 
 |   EXPECT_TRUE(client->default_network_helper()->CreateUDPSocketAndBind( | 
 |       client->server_address(), client->bind_to_address(), | 
 |       client->local_port())); | 
 |   EXPECT_EQ(number_of_open_fds + 2, NumOpenSocketFDs()); | 
 |   EXPECT_TRUE(client->default_network_helper()->CreateUDPSocketAndBind( | 
 |       client->server_address(), client->bind_to_address(), | 
 |       client->local_port())); | 
 |   EXPECT_EQ(number_of_open_fds + 3, NumOpenSocketFDs()); | 
 |  | 
 |   // Clean up UDP sockets. | 
 |   client->default_network_helper()->CleanUpUDPSocket(client->GetLatestFD()); | 
 |   EXPECT_EQ(number_of_open_fds + 2, NumOpenSocketFDs()); | 
 |   client->default_network_helper()->CleanUpUDPSocket(client->GetLatestFD()); | 
 |   EXPECT_EQ(number_of_open_fds + 1, NumOpenSocketFDs()); | 
 | } | 
 |  | 
 | }  // namespace | 
 | }  // namespace test | 
 | }  // namespace quic | 
 |  | 
 | #endif  // defined(__linux__) |