|  | /* | 
|  | *  Copyright (c) 2021 The WebRTC project authors. All Rights Reserved. | 
|  | * | 
|  | *  Use of this source code is governed by a BSD-style license | 
|  | *  that can be found in the LICENSE file in the root of the source | 
|  | *  tree. An additional intellectual property rights grant can be found | 
|  | *  in the file PATENTS.  All contributing project authors may | 
|  | *  be found in the AUTHORS file in the root of the source tree. | 
|  | */ | 
|  | #include "net/dcsctp/socket/heartbeat_handler.h" | 
|  |  | 
|  | #include <cstdint> | 
|  | #include <optional> | 
|  | #include <utility> | 
|  | #include <vector> | 
|  |  | 
|  | #include "api/task_queue/task_queue_base.h" | 
|  | #include "api/units/time_delta.h" | 
|  | #include "net/dcsctp/packet/chunk/heartbeat_ack_chunk.h" | 
|  | #include "net/dcsctp/packet/chunk/heartbeat_request_chunk.h" | 
|  | #include "net/dcsctp/packet/parameter/heartbeat_info_parameter.h" | 
|  | #include "net/dcsctp/packet/parameter/parameter.h" | 
|  | #include "net/dcsctp/packet/sctp_packet.h" | 
|  | #include "net/dcsctp/public/dcsctp_options.h" | 
|  | #include "net/dcsctp/public/types.h" | 
|  | #include "net/dcsctp/socket/mock_context.h" | 
|  | #include "net/dcsctp/socket/mock_dcsctp_socket_callbacks.h" | 
|  | #include "net/dcsctp/testing/testing_macros.h" | 
|  | #include "net/dcsctp/timer/timer.h" | 
|  | #include "test/gmock.h" | 
|  | #include "test/gtest.h" | 
|  |  | 
|  | namespace dcsctp { | 
|  | namespace { | 
|  | using ::testing::ElementsAre; | 
|  | using ::testing::IsEmpty; | 
|  | using ::testing::NiceMock; | 
|  | using ::testing::Return; | 
|  | using ::testing::SizeIs; | 
|  | using ::webrtc::TimeDelta; | 
|  |  | 
|  | constexpr DurationMs kHeartbeatInterval = DurationMs(30'000); | 
|  |  | 
|  | DcSctpOptions MakeOptions(DurationMs heartbeat_interval) { | 
|  | DcSctpOptions options; | 
|  | options.heartbeat_interval_include_rtt = false; | 
|  | options.heartbeat_interval = heartbeat_interval; | 
|  | options.zero_checksum_alternate_error_detection_method = | 
|  | ZeroChecksumAlternateErrorDetectionMethod::None(); | 
|  | return options; | 
|  | } | 
|  |  | 
|  | class HeartbeatHandlerTestBase : public testing::Test { | 
|  | protected: | 
|  | explicit HeartbeatHandlerTestBase(DurationMs heartbeat_interval) | 
|  | : options_(MakeOptions(heartbeat_interval)), | 
|  | context_(&callbacks_), | 
|  | timer_manager_([this](webrtc::TaskQueueBase::DelayPrecision precision) { | 
|  | return callbacks_.CreateTimeout(precision); | 
|  | }), | 
|  | handler_("log: ", options_, &context_, &timer_manager_) {} | 
|  |  | 
|  | void AdvanceTime(webrtc::TimeDelta duration) { | 
|  | callbacks_.AdvanceTime(duration); | 
|  | for (;;) { | 
|  | std::optional<TimeoutID> timeout_id = callbacks_.GetNextExpiredTimeout(); | 
|  | if (!timeout_id.has_value()) { | 
|  | break; | 
|  | } | 
|  | timer_manager_.HandleTimeout(*timeout_id); | 
|  | } | 
|  | } | 
|  |  | 
|  | const DcSctpOptions options_; | 
|  | NiceMock<MockDcSctpSocketCallbacks> callbacks_; | 
|  | NiceMock<MockContext> context_; | 
|  | TimerManager timer_manager_; | 
|  | HeartbeatHandler handler_; | 
|  | }; | 
|  |  | 
|  | class HeartbeatHandlerTest : public HeartbeatHandlerTestBase { | 
|  | protected: | 
|  | HeartbeatHandlerTest() : HeartbeatHandlerTestBase(kHeartbeatInterval) {} | 
|  | }; | 
|  |  | 
|  | class DisabledHeartbeatHandlerTest : public HeartbeatHandlerTestBase { | 
|  | protected: | 
|  | DisabledHeartbeatHandlerTest() : HeartbeatHandlerTestBase(DurationMs(0)) {} | 
|  | }; | 
|  |  | 
|  | TEST_F(HeartbeatHandlerTest, HasRunningHeartbeatIntervalTimer) { | 
|  | AdvanceTime(options_.heartbeat_interval.ToTimeDelta()); | 
|  |  | 
|  | // Validate that a heartbeat request was sent. | 
|  | std::vector<uint8_t> payload = callbacks_.ConsumeSentPacket(); | 
|  | ASSERT_HAS_VALUE_AND_ASSIGN(SctpPacket packet, | 
|  | SctpPacket::Parse(payload, options_)); | 
|  | ASSERT_THAT(packet.descriptors(), SizeIs(1)); | 
|  |  | 
|  | ASSERT_HAS_VALUE_AND_ASSIGN( | 
|  | HeartbeatRequestChunk request, | 
|  | HeartbeatRequestChunk::Parse(packet.descriptors()[0].data)); | 
|  |  | 
|  | EXPECT_TRUE(request.info().has_value()); | 
|  | } | 
|  |  | 
|  | TEST_F(HeartbeatHandlerTest, RepliesToHeartbeatRequests) { | 
|  | uint8_t info_data[] = {1, 2, 3, 4, 5}; | 
|  | HeartbeatRequestChunk request( | 
|  | Parameters::Builder().Add(HeartbeatInfoParameter(info_data)).Build()); | 
|  |  | 
|  | handler_.HandleHeartbeatRequest(std::move(request)); | 
|  |  | 
|  | std::vector<uint8_t> payload = callbacks_.ConsumeSentPacket(); | 
|  | ASSERT_HAS_VALUE_AND_ASSIGN(SctpPacket packet, | 
|  | SctpPacket::Parse(payload, options_)); | 
|  | ASSERT_THAT(packet.descriptors(), SizeIs(1)); | 
|  |  | 
|  | ASSERT_HAS_VALUE_AND_ASSIGN( | 
|  | HeartbeatAckChunk response, | 
|  | HeartbeatAckChunk::Parse(packet.descriptors()[0].data)); | 
|  |  | 
|  | ASSERT_HAS_VALUE_AND_ASSIGN( | 
|  | HeartbeatInfoParameter param, | 
|  | response.parameters().get<HeartbeatInfoParameter>()); | 
|  |  | 
|  | EXPECT_THAT(param.info(), ElementsAre(1, 2, 3, 4, 5)); | 
|  | } | 
|  |  | 
|  | TEST_F(HeartbeatHandlerTest, SendsHeartbeatRequestsOnIdleChannel) { | 
|  | AdvanceTime(options_.heartbeat_interval.ToTimeDelta()); | 
|  |  | 
|  | // Grab the request, and make a response. | 
|  | std::vector<uint8_t> payload = callbacks_.ConsumeSentPacket(); | 
|  | ASSERT_HAS_VALUE_AND_ASSIGN(SctpPacket packet, | 
|  | SctpPacket::Parse(payload, options_)); | 
|  | ASSERT_THAT(packet.descriptors(), SizeIs(1)); | 
|  |  | 
|  | ASSERT_HAS_VALUE_AND_ASSIGN( | 
|  | HeartbeatRequestChunk req, | 
|  | HeartbeatRequestChunk::Parse(packet.descriptors()[0].data)); | 
|  |  | 
|  | HeartbeatAckChunk ack(std::move(req).extract_parameters()); | 
|  |  | 
|  | // Respond a while later. This RTT will be measured by the handler | 
|  | constexpr TimeDelta rtt = TimeDelta::Millis(313); | 
|  |  | 
|  | EXPECT_CALL(context_, ObserveRTT(rtt)).Times(1); | 
|  |  | 
|  | callbacks_.AdvanceTime(rtt); | 
|  | handler_.HandleHeartbeatAck(std::move(ack)); | 
|  | } | 
|  |  | 
|  | TEST_F(HeartbeatHandlerTest, DoesntObserveInvalidHeartbeats) { | 
|  | AdvanceTime(options_.heartbeat_interval.ToTimeDelta()); | 
|  |  | 
|  | // Grab the request, and make a response. | 
|  | std::vector<uint8_t> payload = callbacks_.ConsumeSentPacket(); | 
|  | ASSERT_HAS_VALUE_AND_ASSIGN(SctpPacket packet, | 
|  | SctpPacket::Parse(payload, options_)); | 
|  | ASSERT_THAT(packet.descriptors(), SizeIs(1)); | 
|  |  | 
|  | ASSERT_HAS_VALUE_AND_ASSIGN( | 
|  | HeartbeatRequestChunk req, | 
|  | HeartbeatRequestChunk::Parse(packet.descriptors()[0].data)); | 
|  |  | 
|  | HeartbeatAckChunk ack(std::move(req).extract_parameters()); | 
|  |  | 
|  | EXPECT_CALL(context_, ObserveRTT).Times(0); | 
|  |  | 
|  | // Go backwards in time - which make the HEARTBEAT-ACK have an invalid | 
|  | // timestamp in it, as it will be in the future. | 
|  | callbacks_.AdvanceTime(TimeDelta::Millis(-100)); | 
|  |  | 
|  | handler_.HandleHeartbeatAck(std::move(ack)); | 
|  | } | 
|  |  | 
|  | TEST_F(HeartbeatHandlerTest, IncreasesErrorIfNotAckedInTime) { | 
|  | TimeDelta rto = TimeDelta::Millis(105); | 
|  | EXPECT_CALL(context_, current_rto).WillOnce(Return(rto)); | 
|  | AdvanceTime(options_.heartbeat_interval.ToTimeDelta()); | 
|  |  | 
|  | // Validate that a request was sent. | 
|  | EXPECT_THAT(callbacks_.ConsumeSentPacket(), Not(IsEmpty())); | 
|  |  | 
|  | EXPECT_CALL(context_, IncrementTxErrorCounter).Times(1); | 
|  | AdvanceTime(rto); | 
|  | } | 
|  |  | 
|  | TEST_F(DisabledHeartbeatHandlerTest, IsReallyDisabled) { | 
|  | AdvanceTime(options_.heartbeat_interval.ToTimeDelta()); | 
|  |  | 
|  | // Validate that a request was NOT sent. | 
|  | EXPECT_THAT(callbacks_.ConsumeSentPacket(), IsEmpty()); | 
|  | } | 
|  |  | 
|  | }  // namespace | 
|  | }  // namespace dcsctp |