Adds a wrapper class for the nghttp2 session data type.
PiperOrigin-RevId: 367513350
Change-Id: If86cb1e72347fef86ed6df76a1723cdaf0b4584a
diff --git a/http2/adapter/http2_adapter.h b/http2/adapter/http2_adapter.h
index 5ea6045..4ce8a48 100644
--- a/http2/adapter/http2_adapter.h
+++ b/http2/adapter/http2_adapter.h
@@ -19,11 +19,6 @@
// implementations.
class Http2Adapter {
public:
- enum class Perspective {
- kClient,
- kServer,
- };
-
Http2Adapter(const Http2Adapter&) = delete;
Http2Adapter& operator=(const Http2Adapter&) = delete;
diff --git a/http2/adapter/http2_session.h b/http2/adapter/http2_session.h
index ddbed44..43a26e9 100644
--- a/http2/adapter/http2_session.h
+++ b/http2/adapter/http2_session.h
@@ -26,13 +26,9 @@
virtual int GetRemoteWindowSize() const = 0;
};
-class Http2Options {
- public:
- Http2Options() = default;
- virtual ~Http2Options() {}
-
- // This method returns an opaque reference to the underlying type.
- virtual void* GetOptions() = 0;
+enum class Perspective {
+ kClient,
+ kServer,
};
} // namespace adapter
diff --git a/http2/adapter/nghttp2_callbacks.cc b/http2/adapter/nghttp2_callbacks.cc
index 337f230..e897fa8 100644
--- a/http2/adapter/nghttp2_callbacks.cc
+++ b/http2/adapter/nghttp2_callbacks.cc
@@ -30,6 +30,8 @@
void* user_data) {
auto* visitor = static_cast<Http2VisitorInterface*>(user_data);
const Http2StreamId stream_id = frame->hd.stream_id;
+ QUICHE_VLOG(2) << "Frame " << static_cast<int>(frame->hd.type)
+ << " for stream " << stream_id;
switch (frame->hd.type) {
// The beginning of the DATA frame is handled in OnBeginFrame(), and the
// beginning of the header block is handled in client/server-specific
diff --git a/http2/adapter/nghttp2_session.cc b/http2/adapter/nghttp2_session.cc
new file mode 100644
index 0000000..b288037
--- /dev/null
+++ b/http2/adapter/nghttp2_session.cc
@@ -0,0 +1,73 @@
+#include "http2/adapter/nghttp2_session.h"
+
+#include "common/platform/api/quiche_logging.h"
+
+namespace http2 {
+namespace adapter {
+namespace {
+
+void DeleteSession(nghttp2_session* session) {
+ nghttp2_session_del(session);
+}
+
+void DeleteOptions(nghttp2_option* options) {
+ nghttp2_option_del(options);
+}
+
+} // namespace
+
+NgHttp2Session::NgHttp2Session(Perspective perspective,
+ nghttp2_session_callbacks* callbacks,
+ nghttp2_option* options,
+ void* userdata)
+ : session_(nullptr, DeleteSession),
+ options_(options, DeleteOptions),
+ perspective_(perspective) {
+ nghttp2_session* session;
+ switch (perspective) {
+ case Perspective::kClient:
+ nghttp2_session_client_new2(&session, callbacks, userdata,
+ options_.get());
+ break;
+ case Perspective::kServer:
+ nghttp2_session_server_new2(&session, callbacks, userdata,
+ options_.get());
+ break;
+ }
+ nghttp2_session_callbacks_del(callbacks);
+ session_.reset(session);
+}
+
+NgHttp2Session::~NgHttp2Session() {
+ // Can't invoke want_read() or want_write(), as they are virtual methods.
+ const bool pending_reads = nghttp2_session_want_read(session_.get()) != 0;
+ const bool pending_writes = nghttp2_session_want_write(session_.get()) != 0;
+ QUICHE_LOG_IF(WARNING, pending_reads || pending_writes)
+ << "Shutting down connection with pending reads: " << pending_reads
+ << " or pending writes: " << pending_writes;
+}
+
+ssize_t NgHttp2Session::ProcessBytes(absl::string_view bytes) {
+ return nghttp2_session_mem_recv(
+ session_.get(), reinterpret_cast<const uint8_t*>(bytes.data()),
+ bytes.size());
+}
+
+int NgHttp2Session::Consume(Http2StreamId stream_id, size_t num_bytes) {
+ return nghttp2_session_consume(session_.get(), stream_id, num_bytes);
+}
+
+bool NgHttp2Session::want_read() const {
+ return nghttp2_session_want_read(session_.get()) != 0;
+}
+
+bool NgHttp2Session::want_write() const {
+ return nghttp2_session_want_write(session_.get()) != 0;
+}
+
+int NgHttp2Session::GetRemoteWindowSize() const {
+ return nghttp2_session_get_remote_window_size(session_.get());
+}
+
+} // namespace adapter
+} // namespace http2
diff --git a/http2/adapter/nghttp2_session.h b/http2/adapter/nghttp2_session.h
new file mode 100644
index 0000000..27a2153
--- /dev/null
+++ b/http2/adapter/nghttp2_session.h
@@ -0,0 +1,41 @@
+#ifndef QUICHE_HTTP2_ADAPTER_NGHTTP2_SESSION_H_
+#define QUICHE_HTTP2_ADAPTER_NGHTTP2_SESSION_H_
+
+#include "http2/adapter/http2_session.h"
+#include "third_party/nghttp2/src/lib/includes/nghttp2/nghttp2.h"
+
+namespace http2 {
+namespace adapter {
+
+// A C++ wrapper around common nghttp2_session operations.
+class NgHttp2Session : public Http2Session {
+ public:
+ NgHttp2Session(Perspective perspective,
+ nghttp2_session_callbacks* callbacks,
+ nghttp2_option* options,
+ void* userdata);
+ ~NgHttp2Session() override;
+
+ ssize_t ProcessBytes(absl::string_view bytes) override;
+
+ int Consume(Http2StreamId stream_id, size_t num_bytes) override;
+
+ bool want_read() const override;
+ bool want_write() const override;
+ int GetRemoteWindowSize() const override;
+
+ nghttp2_session* raw_ptr() const { return session_.get(); }
+
+ private:
+ using SessionDeleter = void (&)(nghttp2_session*);
+ using OptionsDeleter = void (&)(nghttp2_option*);
+
+ std::unique_ptr<nghttp2_session, SessionDeleter> session_;
+ std::unique_ptr<nghttp2_option, OptionsDeleter> options_;
+ Perspective perspective_;
+};
+
+} // namespace adapter
+} // namespace http2
+
+#endif // QUICHE_HTTP2_ADAPTER_NGHTTP2_SESSION_H_
diff --git a/http2/adapter/nghttp2_session_test.cc b/http2/adapter/nghttp2_session_test.cc
new file mode 100644
index 0000000..ad74013
--- /dev/null
+++ b/http2/adapter/nghttp2_session_test.cc
@@ -0,0 +1,239 @@
+#include "http2/adapter/nghttp2_session.h"
+
+#include "http2/adapter/mock_http2_visitor.h"
+#include "http2/adapter/nghttp2_callbacks.h"
+#include "http2/adapter/nghttp2_util.h"
+#include "http2/adapter/test_frame_sequence.h"
+#include "common/platform/api/quiche_test.h"
+
+namespace http2 {
+namespace adapter {
+namespace test {
+namespace {
+
+class DataSavingVisitor : public testing::StrictMock<MockHttp2Visitor> {
+ public:
+ void Save(absl::string_view data) { absl::StrAppend(&data_, data); }
+
+ const std::string& data() { return data_; }
+
+ private:
+ std::string data_;
+};
+
+ssize_t SaveSessionOutput(nghttp2_session* session,
+ const uint8_t* data,
+ size_t length,
+ int flags,
+ void* user_data) {
+ auto visitor = static_cast<DataSavingVisitor*>(user_data);
+ visitor->Save(ToStringView(data, length));
+ return length;
+}
+
+class NgHttp2SessionTest : public testing::Test {
+ public:
+ nghttp2_option* CreateOptions() {
+ nghttp2_option* options;
+ nghttp2_option_new(&options);
+ nghttp2_option_set_no_auto_window_update(options, 1);
+ return options;
+ }
+
+ nghttp2_session_callbacks* CreateCallbacks() {
+ nghttp2_session_callbacks* callbacks;
+ nghttp2_session_callbacks_new(&callbacks);
+
+ nghttp2_session_callbacks_set_on_begin_frame_callback(callbacks,
+ &OnBeginFrame);
+ nghttp2_session_callbacks_set_on_frame_recv_callback(callbacks,
+ &OnFrameReceived);
+ nghttp2_session_callbacks_set_on_begin_headers_callback(callbacks,
+ &OnBeginHeaders);
+ nghttp2_session_callbacks_set_on_header_callback2(callbacks, &OnHeader);
+ nghttp2_session_callbacks_set_on_data_chunk_recv_callback(callbacks,
+ &OnDataChunk);
+ nghttp2_session_callbacks_set_on_stream_close_callback(callbacks,
+ &OnStreamClosed);
+
+ nghttp2_session_callbacks_set_send_callback(callbacks, &SaveSessionOutput);
+ return callbacks;
+ }
+
+ DataSavingVisitor visitor_;
+};
+
+TEST_F(NgHttp2SessionTest, ClientConstruction) {
+ NgHttp2Session session(Perspective::kClient, CreateCallbacks(),
+ CreateOptions(), &visitor_);
+ EXPECT_TRUE(session.want_read());
+ EXPECT_FALSE(session.want_write());
+ EXPECT_EQ(session.GetRemoteWindowSize(), kDefaultInitialStreamWindowSize);
+ EXPECT_NE(session.raw_ptr(), nullptr);
+}
+
+TEST_F(NgHttp2SessionTest, ClientHandlesFrames) {
+ NgHttp2Session session(Perspective::kClient, CreateCallbacks(),
+ CreateOptions(), &visitor_);
+
+ ASSERT_EQ(0, nghttp2_session_send(session.raw_ptr()));
+ ASSERT_GT(visitor_.data().size(), 0);
+
+ const std::string initial_frames = TestFrameSequence()
+ .ServerPreface()
+ .Ping(42)
+ .WindowUpdate(0, 1000)
+ .Serialize();
+ testing::InSequence s;
+
+ // Server preface (empty SETTINGS)
+ EXPECT_CALL(visitor_, OnSettingsStart());
+ EXPECT_CALL(visitor_, OnSettingsEnd());
+
+ EXPECT_CALL(visitor_, OnPing(42, false));
+ EXPECT_CALL(visitor_, OnWindowUpdate(0, 1000));
+
+ const ssize_t initial_result = session.ProcessBytes(initial_frames);
+ EXPECT_EQ(initial_frames.size(), initial_result);
+
+ EXPECT_EQ(session.GetRemoteWindowSize(),
+ kDefaultInitialStreamWindowSize + 1000);
+ ASSERT_EQ(0, nghttp2_session_send(session.raw_ptr()));
+
+ const std::vector<Header> headers1 = {{":method", "GET"},
+ {":scheme", "http"},
+ {":authority", "example.com"},
+ {":path", "/this/is/request/one"}};
+ const auto nvs1 = GetRequestNghttp2Nvs(headers1);
+
+ const std::vector<Header> headers2 = {{":method", "GET"},
+ {":scheme", "http"},
+ {":authority", "example.com"},
+ {":path", "/this/is/request/two"}};
+ const auto nvs2 = GetRequestNghttp2Nvs(headers2);
+
+ const std::vector<Header> headers3 = {{":method", "GET"},
+ {":scheme", "http"},
+ {":authority", "example.com"},
+ {":path", "/this/is/request/three"}};
+ const auto nvs3 = GetRequestNghttp2Nvs(headers3);
+
+ const int32_t stream_id1 = nghttp2_submit_request(
+ session.raw_ptr(), nullptr, nvs1.data(), nvs1.size(), nullptr, nullptr);
+ ASSERT_GT(stream_id1, 0);
+ QUICHE_LOG(INFO) << "Created stream: " << stream_id1;
+
+ const int32_t stream_id2 = nghttp2_submit_request(
+ session.raw_ptr(), nullptr, nvs2.data(), nvs2.size(), nullptr, nullptr);
+ ASSERT_GT(stream_id2, 0);
+ QUICHE_LOG(INFO) << "Created stream: " << stream_id2;
+
+ const int32_t stream_id3 = nghttp2_submit_request(
+ session.raw_ptr(), nullptr, nvs3.data(), nvs3.size(), nullptr, nullptr);
+ ASSERT_GT(stream_id3, 0);
+ QUICHE_LOG(INFO) << "Created stream: " << stream_id3;
+
+ ASSERT_EQ(0, nghttp2_session_send(session.raw_ptr()));
+
+ const std::string stream_frames =
+ TestFrameSequence()
+ .Headers(1,
+ {{":status", "200"},
+ {"server", "my-fake-server"},
+ {"date", "Tue, 6 Apr 2021 12:54:01 GMT"}},
+ /*fin=*/false)
+ .Data(1, "This is the response body.")
+ .RstStream(3, Http2ErrorCode::INTERNAL_ERROR)
+ .GoAway(5, Http2ErrorCode::ENHANCE_YOUR_CALM, "calm down!!")
+ .Serialize();
+
+ EXPECT_CALL(visitor_, OnBeginHeadersForStream(1));
+ EXPECT_CALL(visitor_, OnHeaderForStream(1, ":status", "200"));
+ EXPECT_CALL(visitor_, OnHeaderForStream(1, "server", "my-fake-server"));
+ EXPECT_CALL(visitor_,
+ OnHeaderForStream(1, "date", "Tue, 6 Apr 2021 12:54:01 GMT"));
+ EXPECT_CALL(visitor_, OnEndHeadersForStream(1));
+ EXPECT_CALL(visitor_, OnBeginDataForStream(1, 26));
+ EXPECT_CALL(visitor_, OnDataForStream(1, "This is the response body."));
+ EXPECT_CALL(visitor_, OnRstStream(3, Http2ErrorCode::INTERNAL_ERROR));
+ EXPECT_CALL(visitor_, OnAbortStream(3, Http2ErrorCode::INTERNAL_ERROR));
+ EXPECT_CALL(visitor_,
+ OnGoAway(5, Http2ErrorCode::ENHANCE_YOUR_CALM, "calm down!!"));
+ const ssize_t stream_result = session.ProcessBytes(stream_frames);
+ EXPECT_EQ(stream_frames.size(), stream_result);
+ ASSERT_EQ(0, nghttp2_session_send(session.raw_ptr()));
+}
+
+TEST_F(NgHttp2SessionTest, ServerConstruction) {
+ NgHttp2Session session(Perspective::kServer, CreateCallbacks(),
+ CreateOptions(), &visitor_);
+ EXPECT_TRUE(session.want_read());
+ EXPECT_FALSE(session.want_write());
+ EXPECT_EQ(session.GetRemoteWindowSize(), kDefaultInitialStreamWindowSize);
+ EXPECT_NE(session.raw_ptr(), nullptr);
+}
+
+TEST_F(NgHttp2SessionTest, ServerHandlesFrames) {
+ NgHttp2Session session(Perspective::kServer, CreateCallbacks(),
+ CreateOptions(), &visitor_);
+
+ const std::string frames = TestFrameSequence()
+ .ClientPreface()
+ .Ping(42)
+ .WindowUpdate(0, 1000)
+ .Headers(1,
+ {{":method", "POST"},
+ {":scheme", "https"},
+ {":authority", "example.com"},
+ {":path", "/this/is/request/one"}},
+ /*fin=*/false)
+ .WindowUpdate(1, 2000)
+ .Data(1, "This is the request body.")
+ .Headers(3,
+ {{":method", "GET"},
+ {":scheme", "http"},
+ {":authority", "example.com"},
+ {":path", "/this/is/request/two"}},
+ /*fin=*/true)
+ .RstStream(3, Http2ErrorCode::CANCEL)
+ .Ping(47)
+ .Serialize();
+ testing::InSequence s;
+
+ // Client preface (empty SETTINGS)
+ EXPECT_CALL(visitor_, OnSettingsStart());
+ EXPECT_CALL(visitor_, OnSettingsEnd());
+
+ EXPECT_CALL(visitor_, OnPing(42, false));
+ EXPECT_CALL(visitor_, OnWindowUpdate(0, 1000));
+ EXPECT_CALL(visitor_, OnBeginHeadersForStream(1));
+ EXPECT_CALL(visitor_, OnHeaderForStream(1, ":method", "POST"));
+ EXPECT_CALL(visitor_, OnHeaderForStream(1, ":scheme", "https"));
+ EXPECT_CALL(visitor_, OnHeaderForStream(1, ":authority", "example.com"));
+ EXPECT_CALL(visitor_, OnHeaderForStream(1, ":path", "/this/is/request/one"));
+ EXPECT_CALL(visitor_, OnEndHeadersForStream(1));
+ EXPECT_CALL(visitor_, OnWindowUpdate(1, 2000));
+ EXPECT_CALL(visitor_, OnBeginDataForStream(1, 25));
+ EXPECT_CALL(visitor_, OnDataForStream(1, "This is the request body."));
+ EXPECT_CALL(visitor_, OnBeginHeadersForStream(3));
+ EXPECT_CALL(visitor_, OnHeaderForStream(3, ":method", "GET"));
+ EXPECT_CALL(visitor_, OnHeaderForStream(3, ":scheme", "http"));
+ EXPECT_CALL(visitor_, OnHeaderForStream(3, ":authority", "example.com"));
+ EXPECT_CALL(visitor_, OnHeaderForStream(3, ":path", "/this/is/request/two"));
+ EXPECT_CALL(visitor_, OnEndHeadersForStream(3));
+ EXPECT_CALL(visitor_, OnEndStream(3));
+ EXPECT_CALL(visitor_, OnRstStream(3, Http2ErrorCode::CANCEL));
+ EXPECT_CALL(visitor_, OnAbortStream(3, Http2ErrorCode::CANCEL));
+ EXPECT_CALL(visitor_, OnPing(47, false));
+
+ const ssize_t result = session.ProcessBytes(frames);
+ EXPECT_EQ(frames.size(), result);
+
+ EXPECT_EQ(session.GetRemoteWindowSize(),
+ kDefaultInitialStreamWindowSize + 1000);
+}
+
+} // namespace
+} // namespace test
+} // namespace adapter
+} // namespace http2
diff --git a/http2/adapter/nghttp2_util.cc b/http2/adapter/nghttp2_util.cc
index e134148..c191f4d 100644
--- a/http2/adapter/nghttp2_util.cc
+++ b/http2/adapter/nghttp2_util.cc
@@ -25,6 +25,10 @@
return absl::string_view(reinterpret_cast<const char*>(pointer), length);
}
+absl::string_view ToStringView(const uint8_t* pointer, size_t length) {
+ return absl::string_view(reinterpret_cast<const char*>(pointer), length);
+}
+
std::vector<nghttp2_nv> GetRequestNghttp2Nvs(absl::Span<const Header> headers) {
const int num_headers = headers.size();
auto nghttp2_nvs = std::vector<nghttp2_nv>(num_headers);
diff --git a/http2/adapter/nghttp2_util.h b/http2/adapter/nghttp2_util.h
index 3fcbfcd..b9c03bb 100644
--- a/http2/adapter/nghttp2_util.h
+++ b/http2/adapter/nghttp2_util.h
@@ -25,6 +25,7 @@
absl::string_view ToStringView(nghttp2_rcbuf* rc_buffer);
absl::string_view ToStringView(uint8_t* pointer, size_t length);
+absl::string_view ToStringView(const uint8_t* pointer, size_t length);
// Returns the nghttp2 header structure from the given request |headers|, which
// must have the correct pseudoheaders preceding other headers.