|  | #include "http2/adapter/nghttp2_data_provider.h" | 
|  |  | 
|  | #include "http2/adapter/test_utils.h" | 
|  | #include "common/platform/api/quiche_test.h" | 
|  |  | 
|  | namespace http2 { | 
|  | namespace adapter { | 
|  | namespace test { | 
|  |  | 
|  | const size_t kFrameHeaderSize = 9; | 
|  |  | 
|  | // Verifies that a nghttp2_data_provider derived from a DataFrameSource works | 
|  | // correctly with nghttp2-style callbacks when the amount of data read is less | 
|  | // than what the source provides. | 
|  | TEST(DataProviderTest, ReadLessThanSourceProvides) { | 
|  | DataSavingVisitor visitor; | 
|  | TestDataFrameSource source(visitor, true); | 
|  | source.AppendPayload("Example payload"); | 
|  | source.EndData(); | 
|  | auto provider = MakeDataProvider(&source); | 
|  | uint32_t data_flags = 0; | 
|  | const int32_t kStreamId = 1; | 
|  | const size_t kReadLength = 10; | 
|  | // Read callback selects a payload length given an upper bound. | 
|  | ssize_t result = | 
|  | provider->read_callback(nullptr, kStreamId, nullptr, kReadLength, | 
|  | &data_flags, &provider->source, nullptr); | 
|  | ASSERT_EQ(kReadLength, result); | 
|  | EXPECT_EQ(NGHTTP2_DATA_FLAG_NO_COPY, data_flags); | 
|  |  | 
|  | const uint8_t framehd[kFrameHeaderSize] = {1, 2, 3, 4, 5, 6, 7, 8, 9}; | 
|  | // Sends the frame header and some payload bytes. | 
|  | int send_result = callbacks::DataFrameSourceSendCallback( | 
|  | nullptr, nullptr, framehd, result, &provider->source, nullptr); | 
|  | EXPECT_EQ(0, send_result); | 
|  | // Data accepted by the visitor includes a frame header and kReadLength bytes | 
|  | // of payload. | 
|  | EXPECT_EQ(visitor.data().size(), kFrameHeaderSize + kReadLength); | 
|  | } | 
|  |  | 
|  | // Verifies that a nghttp2_data_provider derived from a DataFrameSource works | 
|  | // correctly with nghttp2-style callbacks when the amount of data read is more | 
|  | // than what the source provides. | 
|  | TEST(DataProviderTest, ReadMoreThanSourceProvides) { | 
|  | DataSavingVisitor visitor; | 
|  | const absl::string_view kPayload = "Example payload"; | 
|  | TestDataFrameSource source(visitor, true); | 
|  | source.AppendPayload(kPayload); | 
|  | source.EndData(); | 
|  | auto provider = MakeDataProvider(&source); | 
|  | uint32_t data_flags = 0; | 
|  | const int32_t kStreamId = 1; | 
|  | const size_t kReadLength = 30; | 
|  | // Read callback selects a payload length given an upper bound. | 
|  | ssize_t result = | 
|  | provider->read_callback(nullptr, kStreamId, nullptr, kReadLength, | 
|  | &data_flags, &provider->source, nullptr); | 
|  | ASSERT_EQ(kPayload.size(), result); | 
|  | EXPECT_EQ(NGHTTP2_DATA_FLAG_NO_COPY | NGHTTP2_DATA_FLAG_EOF, data_flags); | 
|  |  | 
|  | const uint8_t framehd[kFrameHeaderSize] = {1, 2, 3, 4, 5, 6, 7, 8, 9}; | 
|  | // Sends the frame header and some payload bytes. | 
|  | int send_result = callbacks::DataFrameSourceSendCallback( | 
|  | nullptr, nullptr, framehd, result, &provider->source, nullptr); | 
|  | EXPECT_EQ(0, send_result); | 
|  | // Data accepted by the visitor includes a frame header and the entire | 
|  | // payload. | 
|  | EXPECT_EQ(visitor.data().size(), kFrameHeaderSize + kPayload.size()); | 
|  | } | 
|  |  | 
|  | // Verifies that a nghttp2_data_provider derived from a DataFrameSource works | 
|  | // correctly with nghttp2-style callbacks when the source is blocked. | 
|  | TEST(DataProviderTest, ReadFromBlockedSource) { | 
|  | DataSavingVisitor visitor; | 
|  | // Source has no payload, but also no fin, so it's blocked. | 
|  | TestDataFrameSource source(visitor, false); | 
|  | auto provider = MakeDataProvider(&source); | 
|  | uint32_t data_flags = 0; | 
|  | const int32_t kStreamId = 1; | 
|  | const size_t kReadLength = 10; | 
|  | ssize_t result = | 
|  | provider->read_callback(nullptr, kStreamId, nullptr, kReadLength, | 
|  | &data_flags, &provider->source, nullptr); | 
|  | // Read operation is deferred, since the source is blocked. | 
|  | EXPECT_EQ(NGHTTP2_ERR_DEFERRED, result); | 
|  | } | 
|  |  | 
|  | // Verifies that a nghttp2_data_provider derived from a DataFrameSource works | 
|  | // correctly with nghttp2-style callbacks when the source provides only fin and | 
|  | // no data. | 
|  | TEST(DataProviderTest, ReadFromZeroLengthSource) { | 
|  | DataSavingVisitor visitor; | 
|  | // Empty payload and fin=true indicates the source is done. | 
|  | TestDataFrameSource source(visitor, true); | 
|  | source.EndData(); | 
|  | auto provider = MakeDataProvider(&source); | 
|  | uint32_t data_flags = 0; | 
|  | const int32_t kStreamId = 1; | 
|  | const size_t kReadLength = 10; | 
|  | ssize_t result = | 
|  | provider->read_callback(nullptr, kStreamId, nullptr, kReadLength, | 
|  | &data_flags, &provider->source, nullptr); | 
|  | ASSERT_EQ(0, result); | 
|  | EXPECT_EQ(NGHTTP2_DATA_FLAG_NO_COPY | NGHTTP2_DATA_FLAG_EOF, data_flags); | 
|  |  | 
|  | const uint8_t framehd[kFrameHeaderSize] = {1, 2, 3, 4, 5, 6, 7, 8, 9}; | 
|  | int send_result = callbacks::DataFrameSourceSendCallback( | 
|  | nullptr, nullptr, framehd, result, &provider->source, nullptr); | 
|  | EXPECT_EQ(0, send_result); | 
|  | // Data accepted by the visitor includes a frame header with fin and zero | 
|  | // bytes of payload. | 
|  | EXPECT_EQ(visitor.data().size(), kFrameHeaderSize); | 
|  | } | 
|  |  | 
|  | }  // namespace test | 
|  | }  // namespace adapter | 
|  | }  // namespace http2 |