// Copyright 2015 The Chromium OS Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include <brillo/streams/fake_stream.h>
#include <vector>
#include <base/callback.h>
#include <base/test/simple_test_clock.h>
#include <brillo/bind_lambda.h>
#include <brillo/message_loops/mock_message_loop.h>
#include <gmock/gmock.h>
#include <gtest/gtest.h>
using testing::AnyNumber;
using testing::InSequence;
using testing::_;
namespace brillo {
class FakeStreamTest : public testing::Test {
public:
void SetUp() override {
mock_loop_.SetAsCurrent();
// Ignore calls to RunOnce().
EXPECT_CALL(mock_loop_, RunOnce(true)).Times(AnyNumber());
}
void CreateStream(Stream::AccessMode mode) {
stream_.reset(new FakeStream{mode, &clock_});
}
// Performs non-blocking read on the stream and returns the read data
// as a string in |out_buffer|. Returns true if the read was successful or
// false when an error occurs. |*eos| is set to true when end of stream is
// reached.
bool ReadString(size_t size_to_read, std::string* out_buffer, bool* eos) {
std::vector<char> data;
data.resize(size_to_read);
size_t size_read = 0;
bool ok = stream_->ReadNonBlocking(data.data(), data.size(), &size_read,
eos, nullptr);
if (ok) {
out_buffer->assign(data.data(), data.data() + size_read);
} else {
out_buffer->clear();
}
return ok;
}
// Writes a string to a stream. Returns the number of bytes written or -1
// in case an error occurred.
int WriteString(const std::string& data) {
size_t written = 0;
if (!stream_->WriteNonBlocking(data.data(), data.size(), &written, nullptr))
return -1;
return static_cast<int>(written);
}
protected:
base::SimpleTestClock clock_;
MockMessageLoop mock_loop_{&clock_};
std::unique_ptr<FakeStream> stream_;
const base::TimeDelta zero_delay;
};
TEST_F(FakeStreamTest, InitReadOnly) {
CreateStream(Stream::AccessMode::READ);
EXPECT_TRUE(stream_->IsOpen());
EXPECT_TRUE(stream_->CanRead());
EXPECT_FALSE(stream_->CanWrite());
EXPECT_FALSE(stream_->CanSeek());
EXPECT_FALSE(stream_->CanGetSize());
EXPECT_EQ(0, stream_->GetSize());
EXPECT_EQ(0, stream_->GetRemainingSize());
EXPECT_EQ(0, stream_->GetPosition());
}
TEST_F(FakeStreamTest, InitWriteOnly) {
CreateStream(Stream::AccessMode::WRITE);
EXPECT_TRUE(stream_->IsOpen());
EXPECT_FALSE(stream_->CanRead());
EXPECT_TRUE(stream_->CanWrite());
EXPECT_FALSE(stream_->CanSeek());
EXPECT_FALSE(stream_->CanGetSize());
EXPECT_EQ(0, stream_->GetSize());
EXPECT_EQ(0, stream_->GetRemainingSize());
EXPECT_EQ(0, stream_->GetPosition());
}
TEST_F(FakeStreamTest, InitReadWrite) {
CreateStream(Stream::AccessMode::READ_WRITE);
EXPECT_TRUE(stream_->IsOpen());
EXPECT_TRUE(stream_->CanRead());
EXPECT_TRUE(stream_->CanWrite());
EXPECT_FALSE(stream_->CanSeek());
EXPECT_FALSE(stream_->CanGetSize());
EXPECT_EQ(0, stream_->GetSize());
EXPECT_EQ(0, stream_->GetRemainingSize());
EXPECT_EQ(0, stream_->GetPosition());
}
TEST_F(FakeStreamTest, ReadEmpty) {
CreateStream(Stream::AccessMode::READ);
std::string data;
bool eos = false;
EXPECT_TRUE(ReadString(100, &data, &eos));
EXPECT_TRUE(eos);
EXPECT_TRUE(data.empty());
}
TEST_F(FakeStreamTest, ReadFullPacket) {
CreateStream(Stream::AccessMode::READ);
stream_->AddReadPacketString({}, "foo");
std::string data;
bool eos = false;
EXPECT_TRUE(ReadString(100, &data, &eos));
EXPECT_FALSE(eos);
EXPECT_EQ("foo", data);
EXPECT_TRUE(ReadString(100, &data, &eos));
EXPECT_TRUE(eos);
EXPECT_TRUE(data.empty());
}
TEST_F(FakeStreamTest, ReadPartialPacket) {
CreateStream(Stream::AccessMode::READ);
stream_->AddReadPacketString({}, "foobar");
std::string data;
bool eos = false;
EXPECT_TRUE(ReadString(3, &data, &eos));
EXPECT_FALSE(eos);
EXPECT_EQ("foo", data);
EXPECT_TRUE(ReadString(100, &data, &eos));
EXPECT_FALSE(eos);
EXPECT_EQ("bar", data);
EXPECT_TRUE(ReadString(100, &data, &eos));
EXPECT_TRUE(eos);
EXPECT_TRUE(data.empty());
}
TEST_F(FakeStreamTest, ReadMultiplePackets) {
CreateStream(Stream::AccessMode::READ);
stream_->AddReadPacketString({}, "foobar");
stream_->AddReadPacketString({}, "baz");
stream_->AddReadPacketString({}, "quux");
std::string data;
bool eos = false;
EXPECT_TRUE(ReadString(100, &data, &eos));
EXPECT_FALSE(eos);
EXPECT_EQ("foobar", data);
EXPECT_TRUE(ReadString(2, &data, &eos));
EXPECT_FALSE(eos);
EXPECT_EQ("ba", data);
EXPECT_TRUE(ReadString(100, &data, &eos));
EXPECT_FALSE(eos);
EXPECT_EQ("z", data);
EXPECT_TRUE(ReadString(100, &data, &eos));
EXPECT_FALSE(eos);
EXPECT_EQ("quux", data);
EXPECT_TRUE(ReadString(100, &data, &eos));
EXPECT_TRUE(eos);
EXPECT_TRUE(data.empty());
stream_->AddReadPacketString({}, "foo-bar");
EXPECT_TRUE(ReadString(100, &data, &eos));
EXPECT_FALSE(eos);
EXPECT_EQ("foo-bar", data);
}
TEST_F(FakeStreamTest, ReadPacketsWithDelay) {
CreateStream(Stream::AccessMode::READ);
stream_->AddReadPacketString({}, "foobar");
stream_->AddReadPacketString(base::TimeDelta::FromSeconds(1), "baz");
std::string data;
bool eos = false;
EXPECT_TRUE(ReadString(100, &data, &eos));
EXPECT_FALSE(eos);
EXPECT_EQ("foobar", data);
EXPECT_TRUE(ReadString(100, &data, &eos));
EXPECT_FALSE(eos);
EXPECT_TRUE(data.empty());
EXPECT_TRUE(ReadString(100, &data, &eos));
EXPECT_FALSE(eos);
EXPECT_TRUE(data.empty());
clock_.Advance(base::TimeDelta::FromSeconds(1));
EXPECT_TRUE(ReadString(100, &data, &eos));
EXPECT_FALSE(eos);
EXPECT_EQ("baz", data);
}
TEST_F(FakeStreamTest, ReadPacketsWithError) {
CreateStream(Stream::AccessMode::READ);
stream_->AddReadPacketString({}, "foobar");
stream_->QueueReadErrorWithMessage(base::TimeDelta::FromSeconds(1),
"Dummy error");
stream_->AddReadPacketString({}, "baz");
std::string data;
bool eos = false;
EXPECT_TRUE(ReadString(100, &data, &eos));
EXPECT_FALSE(eos);
EXPECT_EQ("foobar", data);
EXPECT_TRUE(ReadString(100, &data, &eos));
EXPECT_FALSE(eos);
EXPECT_TRUE(data.empty());
EXPECT_TRUE(ReadString(100, &data, &eos));
EXPECT_FALSE(eos);
EXPECT_TRUE(data.empty());
clock_.Advance(base::TimeDelta::FromSeconds(1));
EXPECT_FALSE(ReadString(100, &data, &eos));
EXPECT_TRUE(ReadString(100, &data, &eos));
EXPECT_FALSE(eos);
EXPECT_EQ("baz", data);
}
TEST_F(FakeStreamTest, WaitForDataRead) {
CreateStream(Stream::AccessMode::READ);
EXPECT_CALL(mock_loop_, PostDelayedTask(_, _, zero_delay)).Times(2);
int call_count = 0;
auto callback = [&call_count](Stream::AccessMode mode) {
call_count++;
EXPECT_EQ(Stream::AccessMode::READ, mode);
};
EXPECT_TRUE(stream_->WaitForData(Stream::AccessMode::READ,
base::Bind(callback), nullptr));
mock_loop_.Run();
EXPECT_EQ(1, call_count);
stream_->AddReadPacketString({}, "foobar");
EXPECT_TRUE(stream_->WaitForData(Stream::AccessMode::READ,
base::Bind(callback), nullptr));
mock_loop_.Run();
EXPECT_EQ(2, call_count);
stream_->ClearReadQueue();
auto one_sec_delay = base::TimeDelta::FromSeconds(1);
stream_->AddReadPacketString(one_sec_delay, "baz");
EXPECT_CALL(mock_loop_, PostDelayedTask(_, _, one_sec_delay)).Times(1);
EXPECT_TRUE(stream_->WaitForData(Stream::AccessMode::READ,
base::Bind(callback), nullptr));
mock_loop_.Run();
EXPECT_EQ(3, call_count);
}
TEST_F(FakeStreamTest, ReadAsync) {
CreateStream(Stream::AccessMode::READ);
std::string input_data = "foobar-baz";
size_t split_pos = input_data.find('-');
auto one_sec_delay = base::TimeDelta::FromSeconds(1);
stream_->AddReadPacketString({}, input_data.substr(0, split_pos));
stream_->AddReadPacketString(one_sec_delay, input_data.substr(split_pos));
{
InSequence seq;
EXPECT_CALL(mock_loop_, PostDelayedTask(_, _, zero_delay)).Times(1);
EXPECT_CALL(mock_loop_, PostDelayedTask(_, _, one_sec_delay)).Times(1);
}
std::vector<char> buffer;
buffer.resize(input_data.size());
int success_count = 0;
int error_count = 0;
auto on_success = [&success_count] { success_count++; };
auto on_failure = [&error_count](const Error* /* error */) { error_count++; };
EXPECT_TRUE(stream_->ReadAllAsync(buffer.data(), buffer.size(),
base::Bind(on_success),
base::Bind(on_failure),
nullptr));
mock_loop_.Run();
EXPECT_EQ(1, success_count);
EXPECT_EQ(0, error_count);
EXPECT_EQ(input_data, (std::string{buffer.begin(), buffer.end()}));
}
TEST_F(FakeStreamTest, WriteEmpty) {
CreateStream(Stream::AccessMode::WRITE);
EXPECT_EQ(-1, WriteString("foo"));
}
TEST_F(FakeStreamTest, WritePartial) {
CreateStream(Stream::AccessMode::WRITE);
stream_->ExpectWritePacketSize({}, 6);
EXPECT_EQ(3, WriteString("foo"));
EXPECT_EQ(3, WriteString("bar"));
EXPECT_EQ(-1, WriteString("baz"));
EXPECT_EQ("foobar", stream_->GetFlushedOutputDataAsString());
}
TEST_F(FakeStreamTest, WriteFullPackets) {
CreateStream(Stream::AccessMode::WRITE);
stream_->ExpectWritePacketSize({}, 3);
EXPECT_EQ(3, WriteString("foo"));
EXPECT_EQ(-1, WriteString("bar"));
stream_->ExpectWritePacketSize({}, 3);
EXPECT_EQ(3, WriteString("bar"));
stream_->ExpectWritePacketSize({}, 3);
EXPECT_EQ(3, WriteString("quux"));
EXPECT_EQ("foobarquu", stream_->GetFlushedOutputDataAsString());
}
TEST_F(FakeStreamTest, WriteAndVerifyData) {
CreateStream(Stream::AccessMode::WRITE);
stream_->ExpectWritePacketString({}, "foo");
stream_->ExpectWritePacketString({}, "bar");
EXPECT_EQ(3, WriteString("foobar"));
EXPECT_EQ(3, WriteString("bar"));
stream_->ExpectWritePacketString({}, "foo");
stream_->ExpectWritePacketString({}, "baz");
EXPECT_EQ(3, WriteString("foobar"));
EXPECT_EQ(-1, WriteString("bar"));
stream_->ExpectWritePacketString({}, "foobar");
EXPECT_EQ(3, WriteString("foo"));
EXPECT_EQ(2, WriteString("ba"));
EXPECT_EQ(-1, WriteString("z"));
}
TEST_F(FakeStreamTest, WriteWithDelay) {
CreateStream(Stream::AccessMode::WRITE);
const auto delay = base::TimeDelta::FromMilliseconds(500);
stream_->ExpectWritePacketSize({}, 3);
stream_->ExpectWritePacketSize(delay, 3);
EXPECT_EQ(3, WriteString("foobar"));
EXPECT_EQ(0, WriteString("bar"));
EXPECT_EQ(0, WriteString("bar"));
clock_.Advance(delay);
EXPECT_EQ(3, WriteString("bar"));
EXPECT_EQ("foobar", stream_->GetFlushedOutputDataAsString());
}
TEST_F(FakeStreamTest, WriteWithError) {
CreateStream(Stream::AccessMode::WRITE);
const auto delay = base::TimeDelta::FromMilliseconds(500);
stream_->ExpectWritePacketSize({}, 3);
stream_->QueueWriteError({});
stream_->ExpectWritePacketSize({}, 3);
stream_->QueueWriteErrorWithMessage(delay, "Dummy message");
stream_->ExpectWritePacketString({}, "foobar");
const std::string data = "foobarbaz";
EXPECT_EQ(3, WriteString(data));
EXPECT_EQ(-1, WriteString(data)); // Simulated error #1.
EXPECT_EQ(3, WriteString(data));
EXPECT_EQ(0, WriteString(data)); // Waiting for data...
clock_.Advance(delay);
EXPECT_EQ(-1, WriteString(data)); // Simulated error #2.
EXPECT_EQ(6, WriteString(data));
EXPECT_EQ(-1, WriteString(data)); // No more data expected.
}
TEST_F(FakeStreamTest, WaitForDataWrite) {
CreateStream(Stream::AccessMode::WRITE);
EXPECT_CALL(mock_loop_, PostDelayedTask(_, _, zero_delay)).Times(2);
int call_count = 0;
auto callback = [&call_count](Stream::AccessMode mode) {
call_count++;
EXPECT_EQ(Stream::AccessMode::WRITE, mode);
};
EXPECT_TRUE(stream_->WaitForData(Stream::AccessMode::WRITE,
base::Bind(callback), nullptr));
mock_loop_.Run();
EXPECT_EQ(1, call_count);
stream_->ExpectWritePacketString({}, "foobar");
EXPECT_TRUE(stream_->WaitForData(Stream::AccessMode::WRITE,
base::Bind(callback), nullptr));
mock_loop_.Run();
EXPECT_EQ(2, call_count);
stream_->ClearWriteQueue();
auto one_sec_delay = base::TimeDelta::FromSeconds(1);
stream_->ExpectWritePacketString(one_sec_delay, "baz");
EXPECT_CALL(mock_loop_, PostDelayedTask(_, _, one_sec_delay)).Times(1);
EXPECT_TRUE(stream_->WaitForData(Stream::AccessMode::WRITE,
base::Bind(callback), nullptr));
mock_loop_.Run();
EXPECT_EQ(3, call_count);
}
TEST_F(FakeStreamTest, WriteAsync) {
CreateStream(Stream::AccessMode::WRITE);
std::string output_data = "foobar-baz";
size_t split_pos = output_data.find('-');
auto one_sec_delay = base::TimeDelta::FromSeconds(1);
stream_->ExpectWritePacketString({}, output_data.substr(0, split_pos));
stream_->ExpectWritePacketString(one_sec_delay,
output_data.substr(split_pos));
{
InSequence seq;
EXPECT_CALL(mock_loop_, PostDelayedTask(_, _, zero_delay)).Times(1);
EXPECT_CALL(mock_loop_, PostDelayedTask(_, _, one_sec_delay)).Times(1);
}
int success_count = 0;
int error_count = 0;
auto on_success = [&success_count] { success_count++; };
auto on_failure = [&error_count](const Error* /* error */) { error_count++; };
EXPECT_TRUE(stream_->WriteAllAsync(output_data.data(), output_data.size(),
base::Bind(on_success),
base::Bind(on_failure),
nullptr));
mock_loop_.Run();
EXPECT_EQ(1, success_count);
EXPECT_EQ(0, error_count);
EXPECT_EQ(output_data, stream_->GetFlushedOutputDataAsString());
}
TEST_F(FakeStreamTest, WaitForDataReadWrite) {
CreateStream(Stream::AccessMode::READ_WRITE);
auto one_sec_delay = base::TimeDelta::FromSeconds(1);
auto two_sec_delay = base::TimeDelta::FromSeconds(2);
int call_count = 0;
auto callback = [&call_count](Stream::AccessMode mode,
Stream::AccessMode expected_mode) {
call_count++;
EXPECT_EQ(static_cast<int>(expected_mode), static_cast<int>(mode));
};
stream_->AddReadPacketString(one_sec_delay, "foo");
stream_->ExpectWritePacketString(two_sec_delay, "bar");
EXPECT_CALL(mock_loop_, PostDelayedTask(_, _, one_sec_delay)).Times(1);
EXPECT_TRUE(stream_->WaitForData(Stream::AccessMode::READ_WRITE,
base::Bind(callback,
Stream::AccessMode::READ),
nullptr));
mock_loop_.Run();
EXPECT_EQ(1, call_count);
// The above step has adjusted the clock by 1 second already.
stream_->ClearReadQueue();
stream_->AddReadPacketString(two_sec_delay, "foo");
EXPECT_CALL(mock_loop_, PostDelayedTask(_, _, one_sec_delay)).Times(1);
EXPECT_TRUE(stream_->WaitForData(Stream::AccessMode::READ_WRITE,
base::Bind(callback,
Stream::AccessMode::WRITE),
nullptr));
mock_loop_.Run();
EXPECT_EQ(2, call_count);
clock_.Advance(one_sec_delay);
EXPECT_CALL(mock_loop_, PostDelayedTask(_, _, zero_delay)).Times(1);
EXPECT_TRUE(stream_->WaitForData(Stream::AccessMode::READ_WRITE,
base::Bind(callback,
Stream::AccessMode::READ_WRITE),
nullptr));
mock_loop_.Run();
EXPECT_EQ(3, call_count);
stream_->ClearReadQueue();
stream_->ClearWriteQueue();
stream_->AddReadPacketString(one_sec_delay, "foo");
stream_->ExpectWritePacketString(one_sec_delay, "bar");
EXPECT_CALL(mock_loop_, PostDelayedTask(_, _, one_sec_delay)).Times(1);
EXPECT_TRUE(stream_->WaitForData(Stream::AccessMode::READ_WRITE,
base::Bind(callback,
Stream::AccessMode::READ_WRITE),
nullptr));
mock_loop_.Run();
EXPECT_EQ(4, call_count);
}
} // namespace brillo