Ed Tanous | 40e9b92 | 2024-09-10 13:50:16 -0700 | [diff] [blame] | 1 | // SPDX-License-Identifier: Apache-2.0 |
| 2 | // SPDX-FileCopyrightText: Copyright OpenBMC Authors |
Ed Tanous | 8f79c5b | 2024-01-30 15:56:37 -0800 | [diff] [blame] | 3 | #include "http/server_sent_event.hpp" |
Ed Tanous | 41fe81c | 2024-09-02 15:08:41 -0700 | [diff] [blame] | 4 | #include "http_request.hpp" |
Ed Tanous | 796ba93 | 2020-08-02 04:29:21 +0000 | [diff] [blame^] | 5 | #include "test_stream.hpp" |
Ed Tanous | 8f79c5b | 2024-01-30 15:56:37 -0800 | [diff] [blame] | 6 | |
Ed Tanous | f0b59af | 2024-03-20 13:38:04 -0700 | [diff] [blame] | 7 | #include <boost/asio/buffer.hpp> |
| 8 | #include <boost/asio/io_context.hpp> |
Ed Tanous | 8d45b9c | 2024-04-08 17:50:10 -0700 | [diff] [blame] | 9 | #include <boost/asio/read.hpp> |
Ed Tanous | 8f79c5b | 2024-01-30 15:56:37 -0800 | [diff] [blame] | 10 | |
Ed Tanous | f0b59af | 2024-03-20 13:38:04 -0700 | [diff] [blame] | 11 | #include <chrono> |
Ed Tanous | 8f79c5b | 2024-01-30 15:56:37 -0800 | [diff] [blame] | 12 | #include <memory> |
| 13 | #include <string> |
| 14 | #include <string_view> |
Ed Tanous | f0b59af | 2024-03-20 13:38:04 -0700 | [diff] [blame] | 15 | #include <utility> |
Ed Tanous | 8f79c5b | 2024-01-30 15:56:37 -0800 | [diff] [blame] | 16 | |
| 17 | #include "gtest/gtest.h" |
| 18 | namespace crow |
| 19 | { |
| 20 | namespace sse_socket |
| 21 | { |
| 22 | |
| 23 | namespace |
| 24 | { |
| 25 | |
| 26 | TEST(ServerSentEvent, SseWorks) |
| 27 | { |
| 28 | boost::asio::io_context io; |
Ed Tanous | 796ba93 | 2020-08-02 04:29:21 +0000 | [diff] [blame^] | 29 | TestStream stream(io); |
| 30 | TestStream out(io); |
Ed Tanous | 8f79c5b | 2024-01-30 15:56:37 -0800 | [diff] [blame] | 31 | stream.connect(out); |
| 32 | |
Ed Tanous | f80a87f | 2024-06-16 12:10:33 -0700 | [diff] [blame] | 33 | Request req; |
| 34 | |
Ed Tanous | 8f79c5b | 2024-01-30 15:56:37 -0800 | [diff] [blame] | 35 | bool openCalled = false; |
Patrick Williams | bd79bce | 2024-08-16 15:22:20 -0400 | [diff] [blame] | 36 | auto openHandler = |
| 37 | [&openCalled](Connection&, const Request& /*handedReq*/) { |
| 38 | openCalled = true; |
| 39 | }; |
Ed Tanous | 8f79c5b | 2024-01-30 15:56:37 -0800 | [diff] [blame] | 40 | bool closeCalled = false; |
| 41 | auto closeHandler = [&closeCalled](Connection&) { closeCalled = true; }; |
| 42 | |
Ed Tanous | 796ba93 | 2020-08-02 04:29:21 +0000 | [diff] [blame^] | 43 | std::shared_ptr<ConnectionImpl<TestStream>> conn = |
| 44 | std::make_shared<ConnectionImpl<TestStream>>(std::move(stream), |
| 45 | openHandler, closeHandler); |
Ed Tanous | f80a87f | 2024-06-16 12:10:33 -0700 | [diff] [blame] | 46 | conn->start(req); |
Ed Tanous | 8f79c5b | 2024-01-30 15:56:37 -0800 | [diff] [blame] | 47 | // Connect |
| 48 | { |
| 49 | constexpr std::string_view expected = |
| 50 | "HTTP/1.1 200 OK\r\n" |
| 51 | "Content-Type: text/event-stream\r\n" |
| 52 | "\r\n"; |
| 53 | |
Ed Tanous | 464924c | 2024-12-20 14:49:45 -0800 | [diff] [blame] | 54 | while (out.str().size() != expected.size() || !openCalled) |
Ed Tanous | 8f79c5b | 2024-01-30 15:56:37 -0800 | [diff] [blame] | 55 | { |
| 56 | io.run_for(std::chrono::milliseconds(1)); |
| 57 | } |
| 58 | |
| 59 | std::string eventContent; |
| 60 | eventContent.resize(expected.size()); |
| 61 | boost::asio::read(out, boost::asio::buffer(eventContent)); |
| 62 | |
| 63 | EXPECT_EQ(eventContent, expected); |
| 64 | EXPECT_TRUE(openCalled); |
| 65 | EXPECT_FALSE(closeCalled); |
| 66 | EXPECT_TRUE(out.str().empty()); |
| 67 | } |
| 68 | // Send one event |
| 69 | { |
Ed Tanous | 6d799e1 | 2024-09-11 14:33:37 -0700 | [diff] [blame] | 70 | conn->sendSseEvent("TestEventId", "TestEventContent"); |
Ed Tanous | 8f79c5b | 2024-01-30 15:56:37 -0800 | [diff] [blame] | 71 | std::string_view expected = "id: TestEventId\n" |
| 72 | "data: TestEventContent\n" |
| 73 | "\n"; |
| 74 | |
| 75 | while (out.str().size() < expected.size()) |
| 76 | { |
| 77 | io.run_for(std::chrono::milliseconds(1)); |
| 78 | } |
| 79 | EXPECT_EQ(out.str(), expected); |
| 80 | |
| 81 | std::string eventContent; |
| 82 | eventContent.resize(expected.size()); |
| 83 | boost::asio::read(out, boost::asio::buffer(eventContent)); |
| 84 | |
| 85 | EXPECT_EQ(eventContent, expected); |
| 86 | EXPECT_TRUE(out.str().empty()); |
| 87 | } |
| 88 | // Send second event |
| 89 | { |
Ed Tanous | 6d799e1 | 2024-09-11 14:33:37 -0700 | [diff] [blame] | 90 | conn->sendSseEvent("TestEventId2", "TestEvent\nContent2"); |
Patrick Williams | bd79bce | 2024-08-16 15:22:20 -0400 | [diff] [blame] | 91 | constexpr std::string_view expected = |
| 92 | "id: TestEventId2\n" |
| 93 | "data: TestEvent\n" |
| 94 | "data: Content2\n" |
| 95 | "\n"; |
Ed Tanous | 8f79c5b | 2024-01-30 15:56:37 -0800 | [diff] [blame] | 96 | |
| 97 | while (out.str().size() < expected.size()) |
| 98 | { |
| 99 | io.run_for(std::chrono::milliseconds(1)); |
| 100 | } |
| 101 | |
| 102 | std::string eventContent; |
| 103 | eventContent.resize(expected.size()); |
| 104 | boost::asio::read(out, boost::asio::buffer(eventContent)); |
| 105 | EXPECT_EQ(eventContent, expected); |
| 106 | EXPECT_TRUE(out.str().empty()); |
| 107 | } |
| 108 | // close the remote |
| 109 | { |
| 110 | out.close(); |
| 111 | while (!closeCalled) |
| 112 | { |
| 113 | io.run_for(std::chrono::milliseconds(1)); |
| 114 | } |
| 115 | } |
| 116 | } |
| 117 | } // namespace |
| 118 | |
| 119 | } // namespace sse_socket |
| 120 | } // namespace crow |