diff --git a/include/envoy/stream_info/stream_info.h b/include/envoy/stream_info/stream_info.h index 6a0bbd05e266..37f80b67eebc 100644 --- a/include/envoy/stream_info/stream_info.h +++ b/include/envoy/stream_info/stream_info.h @@ -25,6 +25,11 @@ namespace Router { class RouteEntry; } // namespace Router +namespace Upstream { +class ClusterInfo; +using ClusterInfoConstSharedPtr = std::shared_ptr; +} // namespace Upstream + namespace StreamInfo { enum ResponseFlag { @@ -520,6 +525,19 @@ class StreamInfo { * @return request headers. */ virtual const Http::RequestHeaderMap* getRequestHeaders() const PURE; + + /** + * @param Upstream Connection's ClusterInfo. + */ + virtual void + setUpstreamClusterInfo(const Upstream::ClusterInfoConstSharedPtr& upstream_cluster_info) PURE; + + /** + * @return Upstream Connection's ClusterInfo. + * This returns an optional to differentiate between unset(absl::nullopt), + * no route or cluster does not exist(nullptr), and set to a valid cluster(not nullptr). + */ + virtual absl::optional upstreamClusterInfo() const PURE; }; } // namespace StreamInfo diff --git a/source/common/http/conn_manager_impl.cc b/source/common/http/conn_manager_impl.cc index 010c3fb9b9fa..19ad7c70173e 100644 --- a/source/common/http/conn_manager_impl.cc +++ b/source/common/http/conn_manager_impl.cc @@ -1406,6 +1406,7 @@ void ConnectionManagerImpl::ActiveStream::refreshCachedRoute() { cached_cluster_info_ = (nullptr == local_cluster) ? nullptr : local_cluster->info(); } + stream_info_.setUpstreamClusterInfo(cached_cluster_info_.value()); refreshCachedTracingCustomTags(); } diff --git a/source/common/stream_info/stream_info_impl.h b/source/common/stream_info/stream_info_impl.h index 00707bf9c6b8..85f3296cb948 100644 --- a/source/common/stream_info/stream_info_impl.h +++ b/source/common/stream_info/stream_info_impl.h @@ -253,6 +253,15 @@ struct StreamInfoImpl : public StreamInfo { << DUMP_MEMBER(health_check_request_) << DUMP_MEMBER(route_name_) << "\n"; } + void setUpstreamClusterInfo( + const Upstream::ClusterInfoConstSharedPtr& upstream_cluster_info) override { + upstream_cluster_info_ = upstream_cluster_info; + } + + absl::optional upstreamClusterInfo() const override { + return upstream_cluster_info_; + } + TimeSource& time_source_; const SystemTime start_time_; const MonotonicTime start_time_monotonic_; @@ -287,6 +296,7 @@ struct StreamInfoImpl : public StreamInfo { const Http::RequestHeaderMap* request_headers_{}; UpstreamTiming upstream_timing_; std::string upstream_transport_failure_reason_; + absl::optional upstream_cluster_info_; }; } // namespace StreamInfo diff --git a/source/common/tcp_proxy/tcp_proxy.cc b/source/common/tcp_proxy/tcp_proxy.cc index 62182d0ebe9c..eca88fda3da8 100644 --- a/source/common/tcp_proxy/tcp_proxy.cc +++ b/source/common/tcp_proxy/tcp_proxy.cc @@ -405,6 +405,7 @@ Network::FilterStatus Filter::initializeUpstreamConnection() { } Upstream::ClusterInfoConstSharedPtr cluster = thread_local_cluster->info(); + getStreamInfo().setUpstreamClusterInfo(cluster); // Check this here because the TCP conn pool will queue our request waiting for a connection that // will never be released. diff --git a/test/common/stream_info/stream_info_impl_test.cc b/test/common/stream_info/stream_info_impl_test.cc index d5434f5052be..e51947352282 100644 --- a/test/common/stream_info/stream_info_impl_test.cc +++ b/test/common/stream_info/stream_info_impl_test.cc @@ -11,6 +11,7 @@ #include "test/common/stream_info/test_int_accessor.h" #include "test/mocks/router/mocks.h" +#include "test/mocks/upstream/cluster_info.h" #include "test/mocks/upstream/mocks.h" #include "gmock/gmock.h" @@ -175,6 +176,12 @@ TEST_F(StreamInfoImplTest, MiscSettersAndGetters) { absl::string_view sni_name = "stubserver.org"; stream_info.setRequestedServerName(sni_name); EXPECT_EQ(std::string(sni_name), stream_info.requestedServerName()); + + EXPECT_EQ(absl::nullopt, stream_info.upstreamClusterInfo()); + Upstream::ClusterInfoConstSharedPtr cluster_info(new NiceMock()); + stream_info.setUpstreamClusterInfo(cluster_info); + EXPECT_NE(absl::nullopt, stream_info.upstreamClusterInfo()); + EXPECT_EQ("fake_cluster", stream_info.upstreamClusterInfo().value()->name()); } } diff --git a/test/common/stream_info/test_util.h b/test/common/stream_info/test_util.h index 81d47b23924e..086f31666152 100644 --- a/test/common/stream_info/test_util.h +++ b/test/common/stream_info/test_util.h @@ -209,6 +209,14 @@ class TestStreamInfo : public StreamInfo::StreamInfo { Event::TimeSystem& timeSystem() { return test_time_.timeSystem(); } + void setUpstreamClusterInfo( + const Upstream::ClusterInfoConstSharedPtr& upstream_cluster_info) override { + upstream_cluster_info_ = upstream_cluster_info; + } + absl::optional upstreamClusterInfo() const override { + return upstream_cluster_info_; + } + SystemTime start_time_; MonotonicTime start_time_monotonic_; @@ -245,6 +253,7 @@ class TestStreamInfo : public StreamInfo::StreamInfo { std::string upstream_transport_failure_reason_; const Http::RequestHeaderMap* request_headers_{}; Envoy::Event::SimulatedTimeSystem test_time_; + absl::optional upstream_cluster_info_{}; }; } // namespace Envoy diff --git a/test/common/tcp_proxy/tcp_proxy_test.cc b/test/common/tcp_proxy/tcp_proxy_test.cc index 80d676f5b97a..cfc8159980a5 100644 --- a/test/common/tcp_proxy/tcp_proxy_test.cc +++ b/test/common/tcp_proxy/tcp_proxy_test.cc @@ -1927,6 +1927,45 @@ TEST_F(TcpProxyRoutingTest, DEPRECATED_FEATURE_TEST(ApplicationProtocols)) { filter_->onNewConnection(); } +class TcpProxyNonDeprecatedConfigRoutingTest : public TcpProxyRoutingTest { +public: + TcpProxyNonDeprecatedConfigRoutingTest() = default; + + void setup() { + const std::string yaml = R"EOF( + stat_prefix: name + cluster: fake_cluster + )EOF"; + + config_.reset(new Config(constructConfigFromYaml(yaml, factory_context_))); + } +}; + +TEST_F(TcpProxyNonDeprecatedConfigRoutingTest, ClusterNameSet) { + setup(); + + initializeFilter(); + + // Port 9999 is within the specified destination port range. + connection_.local_address_ = std::make_shared("1.2.3.4", 9999); + + // Expect filter to try to open a connection to specified cluster. + EXPECT_CALL(factory_context_.cluster_manager_, tcpConnPoolForCluster("fake_cluster", _, _)) + .WillOnce(Return(nullptr)); + absl::optional cluster_info; + EXPECT_CALL(connection_.stream_info_, setUpstreamClusterInfo(_)) + .WillOnce( + Invoke([&cluster_info](const Upstream::ClusterInfoConstSharedPtr& upstream_cluster_info) { + cluster_info = upstream_cluster_info; + })); + EXPECT_CALL(connection_.stream_info_, upstreamClusterInfo()) + .WillOnce(ReturnPointee(&cluster_info)); + + filter_->onNewConnection(); + + EXPECT_EQ(connection_.stream_info_.upstreamClusterInfo().value()->name(), "fake_cluster"); +} + class TcpProxyHashingTest : public testing::Test { public: TcpProxyHashingTest() = default; diff --git a/test/mocks/stream_info/mocks.h b/test/mocks/stream_info/mocks.h index 9a49ad422002..4ca1498133ff 100644 --- a/test/mocks/stream_info/mocks.h +++ b/test/mocks/stream_info/mocks.h @@ -89,6 +89,9 @@ class MockStreamInfo : public StreamInfo { MOCK_METHOD(const std::string&, upstreamTransportFailureReason, (), (const)); MOCK_METHOD(void, setRequestHeaders, (const Http::RequestHeaderMap&)); MOCK_METHOD(const Http::RequestHeaderMap*, getRequestHeaders, (), (const)); + MOCK_METHOD(void, setUpstreamClusterInfo, (const Upstream::ClusterInfoConstSharedPtr&)); + MOCK_METHOD(absl::optional, upstreamClusterInfo, (), + (const)); std::shared_ptr> host_{ new testing::NiceMock()};