mirror of
https://github.com/nestriness/cdc-file-transfer.git
synced 2026-01-30 14:45:37 +02:00
Instead of calling netstat on the remote device to detect available ports, simply call bind with port 0 to bind to any available port. Since the port is not yet known when cdc_rsync_server.exe is called, port forwarding needs to be started AFTER the server reports its port.
328 lines
13 KiB
C++
328 lines
13 KiB
C++
// Copyright 2022 Google LLC
|
|
//
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
// you may not use this file except in compliance with the License.
|
|
// You may obtain a copy of the License at
|
|
//
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
//
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
// See the License for the specific language governing permissions and
|
|
// limitations under the License.
|
|
|
|
#include "common/port_manager.h"
|
|
|
|
#include "absl/strings/match.h"
|
|
#include "common/log.h"
|
|
#include "common/remote_util.h"
|
|
#include "common/status_test_macros.h"
|
|
#include "common/stub_process.h"
|
|
#include "common/testing_clock.h"
|
|
#include "gtest/gtest.h"
|
|
|
|
namespace cdc_ft {
|
|
namespace {
|
|
|
|
constexpr char kUserHost[] = "user@1.2.3.4";
|
|
|
|
constexpr char kGuid[] = "f77bcdfe-368c-4c45-9f01-230c5e7e2132";
|
|
constexpr int kFirstPort = 44450;
|
|
constexpr int kLastPort = 44459;
|
|
constexpr int kNumPorts = kLastPort - kFirstPort + 1;
|
|
|
|
constexpr int kTimeoutSec = 1;
|
|
|
|
constexpr char kWindowsNetstat[] = "netstat -a -n -p tcp";
|
|
constexpr char kLinuxNetstatOrSS[] =
|
|
"which ss2 && ss --numeric --listening --tcp || netstat --numeric "
|
|
"--listening --tcp";
|
|
|
|
constexpr char kWindowsNetstatOutFmt[] =
|
|
"TCP 127.0.0.1:50000 127.0.0.1:%i ESTABLISHED";
|
|
constexpr char kLinuxNetstatOutFmt[] =
|
|
"tcp 0 0 0.0.0.0:%i 0.0.0.0:* LISTEN";
|
|
constexpr char kLinuxSSOutFmt[] =
|
|
"LISTEN 0 128 "
|
|
" 0.0.0.0:%i 0.0.0.0:* ";
|
|
|
|
class PortManagerTest : public ::testing::Test {
|
|
public:
|
|
PortManagerTest()
|
|
: remote_util_(kUserHost, /*verbosity=*/0, /*quiet=*/false,
|
|
&process_factory_,
|
|
/*forward_output_to_log=*/true),
|
|
port_manager_(kGuid, kFirstPort, kLastPort, &process_factory_,
|
|
&remote_util_, &system_clock_, &steady_clock_) {}
|
|
|
|
void SetUp() override {
|
|
Log::Initialize(std::make_unique<ConsoleLog>(LogLevel::kInfo));
|
|
}
|
|
|
|
void TearDown() override { Log::Shutdown(); }
|
|
|
|
protected:
|
|
StubProcessFactory process_factory_;
|
|
TestingSystemClock system_clock_;
|
|
TestingSteadyClock steady_clock_;
|
|
RemoteUtil remote_util_;
|
|
PortManager port_manager_;
|
|
};
|
|
|
|
TEST_F(PortManagerTest, ReservePortSuccess) {
|
|
process_factory_.SetProcessOutput(kWindowsNetstat, "", "", 0);
|
|
process_factory_.SetProcessOutput(kLinuxNetstatOrSS, "", "", 0);
|
|
|
|
absl::StatusOr<int> port =
|
|
port_manager_.ReservePort(kTimeoutSec, ArchType::kLinux_x86_64);
|
|
ASSERT_OK(port);
|
|
EXPECT_EQ(*port, kFirstPort);
|
|
}
|
|
|
|
TEST_F(PortManagerTest, ReservePortAllLocalPortsTaken) {
|
|
std::string local_netstat_out = "";
|
|
for (int port = kFirstPort; port <= kLastPort; ++port) {
|
|
local_netstat_out += absl::StrFormat(kWindowsNetstatOutFmt, port);
|
|
}
|
|
process_factory_.SetProcessOutput(kWindowsNetstat, local_netstat_out, "", 0);
|
|
process_factory_.SetProcessOutput(kLinuxNetstatOrSS, "", "", 0);
|
|
|
|
absl::StatusOr<int> port =
|
|
port_manager_.ReservePort(kTimeoutSec, ArchType::kLinux_x86_64);
|
|
EXPECT_TRUE(absl::IsResourceExhausted(port.status()));
|
|
EXPECT_TRUE(
|
|
absl::StrContains(port.status().message(), "No port available in range"));
|
|
}
|
|
|
|
TEST_F(PortManagerTest, ReservePortAllRemotePortsTaken) {
|
|
std::string remote_netstat_out = "";
|
|
for (int port = kFirstPort; port <= kLastPort; ++port) {
|
|
remote_netstat_out += absl::StrFormat(kLinuxNetstatOutFmt, port);
|
|
}
|
|
process_factory_.SetProcessOutput(kWindowsNetstat, "", "", 0);
|
|
process_factory_.SetProcessOutput(kLinuxNetstatOrSS, remote_netstat_out, "",
|
|
0);
|
|
|
|
absl::StatusOr<int> port =
|
|
port_manager_.ReservePort(kTimeoutSec, ArchType::kLinux_x86_64);
|
|
EXPECT_TRUE(absl::IsResourceExhausted(port.status()));
|
|
EXPECT_TRUE(
|
|
absl::StrContains(port.status().message(), "No port available in range"));
|
|
}
|
|
|
|
TEST_F(PortManagerTest, ReservePortLocalNetstatFails) {
|
|
process_factory_.SetProcessOutput(kWindowsNetstat, "", "", 1);
|
|
process_factory_.SetProcessOutput(kLinuxNetstatOrSS, "", "", 0);
|
|
|
|
absl::StatusOr<int> port =
|
|
port_manager_.ReservePort(kTimeoutSec, ArchType::kLinux_x86_64);
|
|
EXPECT_NOT_OK(port);
|
|
EXPECT_TRUE(absl::StrContains(port.status().message(),
|
|
"Failed to find available local ports"));
|
|
}
|
|
|
|
TEST_F(PortManagerTest, ReservePortRemoteNetstatFails) {
|
|
process_factory_.SetProcessOutput(kWindowsNetstat, "", "", 0);
|
|
process_factory_.SetProcessOutput(kLinuxNetstatOrSS, "", "", 1);
|
|
|
|
absl::StatusOr<int> port =
|
|
port_manager_.ReservePort(kTimeoutSec, ArchType::kLinux_x86_64);
|
|
EXPECT_NOT_OK(port);
|
|
EXPECT_TRUE(absl::StrContains(port.status().message(),
|
|
"Failed to find available remote ports"));
|
|
}
|
|
|
|
TEST_F(PortManagerTest, ReservePortRemoteNetstatTimesOut) {
|
|
process_factory_.SetProcessOutput(kWindowsNetstat, "", "", 0);
|
|
process_factory_.SetProcessNeverExits(kLinuxNetstatOrSS);
|
|
steady_clock_.AutoAdvance(kTimeoutSec * 2 * 1000);
|
|
|
|
absl::StatusOr<int> port =
|
|
port_manager_.ReservePort(kTimeoutSec, ArchType::kLinux_x86_64);
|
|
EXPECT_NOT_OK(port);
|
|
EXPECT_TRUE(absl::IsDeadlineExceeded(port.status()));
|
|
EXPECT_TRUE(absl::StrContains(port.status().message(),
|
|
"Timeout while running netstat"));
|
|
}
|
|
|
|
TEST_F(PortManagerTest, ReservePortMultipleInstances) {
|
|
process_factory_.SetProcessOutput(kWindowsNetstat, "", "", 0);
|
|
process_factory_.SetProcessOutput(kLinuxNetstatOrSS, "", "", 0);
|
|
|
|
PortManager port_manager2(kGuid, kFirstPort, kLastPort, &process_factory_,
|
|
&remote_util_);
|
|
|
|
// Port managers use shared memory, so different instances know about each
|
|
// other. This would even work if |port_manager_| and |port_manager2| belonged
|
|
// to different processes, but we don't test that here.
|
|
EXPECT_EQ(*port_manager_.ReservePort(kTimeoutSec, ArchType::kLinux_x86_64),
|
|
kFirstPort + 0);
|
|
EXPECT_EQ(*port_manager2.ReservePort(kTimeoutSec, ArchType::kLinux_x86_64),
|
|
kFirstPort + 1);
|
|
EXPECT_EQ(*port_manager_.ReservePort(kTimeoutSec, ArchType::kLinux_x86_64),
|
|
kFirstPort + 2);
|
|
EXPECT_EQ(*port_manager2.ReservePort(kTimeoutSec, ArchType::kLinux_x86_64),
|
|
kFirstPort + 3);
|
|
}
|
|
|
|
TEST_F(PortManagerTest, ReservePortReusesPortsInLRUOrder) {
|
|
process_factory_.SetProcessOutput(kWindowsNetstat, "", "", 0);
|
|
process_factory_.SetProcessOutput(kLinuxNetstatOrSS, "", "", 0);
|
|
|
|
for (int n = 0; n < kNumPorts * 2; ++n) {
|
|
EXPECT_EQ(*port_manager_.ReservePort(kTimeoutSec, ArchType::kLinux_x86_64),
|
|
kFirstPort + n % kNumPorts);
|
|
system_clock_.Advance(1000);
|
|
}
|
|
}
|
|
|
|
TEST_F(PortManagerTest, ReleasePort) {
|
|
process_factory_.SetProcessOutput(kWindowsNetstat, "", "", 0);
|
|
process_factory_.SetProcessOutput(kLinuxNetstatOrSS, "", "", 0);
|
|
|
|
absl::StatusOr<int> port =
|
|
port_manager_.ReservePort(kTimeoutSec, ArchType::kLinux_x86_64);
|
|
EXPECT_EQ(*port, kFirstPort);
|
|
EXPECT_OK(port_manager_.ReleasePort(*port));
|
|
port = port_manager_.ReservePort(kTimeoutSec, ArchType::kLinux_x86_64);
|
|
EXPECT_EQ(*port, kFirstPort);
|
|
}
|
|
|
|
TEST_F(PortManagerTest, ReleasePortOnDestruction) {
|
|
process_factory_.SetProcessOutput(kWindowsNetstat, "", "", 0);
|
|
process_factory_.SetProcessOutput(kLinuxNetstatOrSS, "", "", 0);
|
|
|
|
auto port_manager2 = std::make_unique<PortManager>(
|
|
kGuid, kFirstPort, kLastPort, &process_factory_, &remote_util_);
|
|
EXPECT_EQ(*port_manager2->ReservePort(kTimeoutSec, ArchType::kLinux_x86_64),
|
|
kFirstPort + 0);
|
|
EXPECT_EQ(*port_manager_.ReservePort(kTimeoutSec, ArchType::kLinux_x86_64),
|
|
kFirstPort + 1);
|
|
port_manager2.reset();
|
|
EXPECT_EQ(*port_manager_.ReservePort(kTimeoutSec, ArchType::kLinux_x86_64),
|
|
kFirstPort + 0);
|
|
}
|
|
|
|
TEST_F(PortManagerTest, FindAvailableLocalPortsSuccessWindows) {
|
|
// First port is in use.
|
|
std::string local_netstat_out =
|
|
absl::StrFormat(kWindowsNetstatOutFmt, kFirstPort);
|
|
process_factory_.SetProcessOutput(kWindowsNetstat, local_netstat_out, "", 0);
|
|
|
|
absl::StatusOr<std::unordered_set<int>> ports =
|
|
PortManager::FindAvailableLocalPorts(
|
|
kFirstPort, kLastPort, ArchType::kWindows_x86_64, &process_factory_);
|
|
ASSERT_OK(ports);
|
|
EXPECT_EQ(ports->size(), kNumPorts - 1);
|
|
for (int port = kFirstPort + 1; port <= kLastPort; ++port) {
|
|
EXPECT_TRUE(ports->find(port) != ports->end());
|
|
}
|
|
}
|
|
|
|
TEST_F(PortManagerTest, FindAvailableLocalPortsSuccessLinux) {
|
|
// First port is in use.
|
|
std::string local_netstat_out =
|
|
absl::StrFormat(kLinuxNetstatOutFmt, kFirstPort);
|
|
process_factory_.SetProcessOutput(kLinuxNetstatOrSS, local_netstat_out, "",
|
|
0);
|
|
|
|
absl::StatusOr<std::unordered_set<int>> ports =
|
|
PortManager::FindAvailableLocalPorts(
|
|
kFirstPort, kLastPort, ArchType::kLinux_x86_64, &process_factory_);
|
|
ASSERT_OK(ports);
|
|
EXPECT_EQ(ports->size(), kNumPorts - 1);
|
|
for (int port = kFirstPort + 1; port <= kLastPort; ++port) {
|
|
EXPECT_TRUE(ports->find(port) != ports->end());
|
|
}
|
|
}
|
|
|
|
TEST_F(PortManagerTest, FindAvailableLocalPortsFailsNoPorts) {
|
|
// All ports are in use.
|
|
std::string local_netstat_out = "";
|
|
for (int port = kFirstPort; port <= kLastPort; ++port) {
|
|
local_netstat_out += absl::StrFormat(kWindowsNetstatOutFmt, port);
|
|
}
|
|
process_factory_.SetProcessOutput(kWindowsNetstat, local_netstat_out, "", 0);
|
|
|
|
absl::StatusOr<std::unordered_set<int>> ports =
|
|
PortManager::FindAvailableLocalPorts(
|
|
kFirstPort, kLastPort, ArchType::kWindows_x86_64, &process_factory_);
|
|
EXPECT_TRUE(absl::IsResourceExhausted(ports.status()));
|
|
EXPECT_TRUE(absl::StrContains(ports.status().message(),
|
|
"No port available in range"));
|
|
}
|
|
|
|
TEST_F(PortManagerTest, FindAvailableRemotePortsSuccessLinux) {
|
|
// First port is in use.
|
|
std::string remote_netstat_out =
|
|
absl::StrFormat(kLinuxNetstatOutFmt, kFirstPort);
|
|
process_factory_.SetProcessOutput(kLinuxNetstatOrSS, remote_netstat_out, "",
|
|
0);
|
|
|
|
absl::StatusOr<std::unordered_set<int>> ports =
|
|
PortManager::FindAvailableRemotePorts(
|
|
kFirstPort, kLastPort, ArchType::kLinux_x86_64, &process_factory_,
|
|
&remote_util_, kTimeoutSec);
|
|
ASSERT_OK(ports);
|
|
EXPECT_EQ(ports->size(), kNumPorts - 1);
|
|
for (int port = kFirstPort + 1; port <= kLastPort; ++port) {
|
|
EXPECT_TRUE(ports->find(port) != ports->end());
|
|
}
|
|
}
|
|
|
|
TEST_F(PortManagerTest, FindAvailableRemotePortsSuccessLinuxSS) {
|
|
// First port is in use, but reporting is done by SS, not netsta.
|
|
std::string remote_netstat_out = absl::StrFormat(kLinuxSSOutFmt, kFirstPort);
|
|
process_factory_.SetProcessOutput(kLinuxNetstatOrSS, remote_netstat_out, "",
|
|
0);
|
|
|
|
absl::StatusOr<std::unordered_set<int>> ports =
|
|
PortManager::FindAvailableRemotePorts(
|
|
kFirstPort, kLastPort, ArchType::kLinux_x86_64, &process_factory_,
|
|
&remote_util_, kTimeoutSec);
|
|
ASSERT_OK(ports);
|
|
EXPECT_EQ(ports->size(), kNumPorts - 1);
|
|
for (int port = kFirstPort + 1; port <= kLastPort; ++port) {
|
|
EXPECT_TRUE(ports->find(port) != ports->end());
|
|
}
|
|
}
|
|
|
|
TEST_F(PortManagerTest, FindAvailableRemotePortsSuccessWindows) {
|
|
// First port is in use.
|
|
std::string remote_netstat_out =
|
|
absl::StrFormat(kWindowsNetstatOutFmt, kFirstPort);
|
|
process_factory_.SetProcessOutput(kWindowsNetstat, remote_netstat_out, "", 0);
|
|
|
|
absl::StatusOr<std::unordered_set<int>> ports =
|
|
PortManager::FindAvailableRemotePorts(
|
|
kFirstPort, kLastPort, ArchType::kWindows_x86_64, &process_factory_,
|
|
&remote_util_, kTimeoutSec);
|
|
ASSERT_OK(ports);
|
|
EXPECT_EQ(ports->size(), kNumPorts - 1);
|
|
for (int port = kFirstPort + 1; port <= kLastPort; ++port) {
|
|
EXPECT_TRUE(ports->find(port) != ports->end());
|
|
}
|
|
}
|
|
|
|
TEST_F(PortManagerTest, FindAvailableRemotePortsFailsNoPorts) {
|
|
// All ports are in use.
|
|
std::string remote_netstat_out = "";
|
|
for (int port = kFirstPort; port <= kLastPort; ++port) {
|
|
remote_netstat_out += absl::StrFormat(kLinuxNetstatOutFmt, port);
|
|
}
|
|
process_factory_.SetProcessOutput(kLinuxNetstatOrSS, remote_netstat_out, "",
|
|
0);
|
|
|
|
absl::StatusOr<std::unordered_set<int>> ports =
|
|
PortManager::FindAvailableRemotePorts(
|
|
kFirstPort, kLastPort, ArchType::kLinux_x86_64, &process_factory_,
|
|
&remote_util_, kTimeoutSec);
|
|
EXPECT_TRUE(absl::IsResourceExhausted(ports.status()));
|
|
EXPECT_TRUE(absl::StrContains(ports.status().message(),
|
|
"No port available in range"));
|
|
}
|
|
|
|
} // namespace
|
|
} // namespace cdc_ft
|