// 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/input_stream_set.h>
#include <brillo/errors/error_codes.h>
#include <brillo/streams/mock_stream.h>
#include <brillo/streams/stream_errors.h>
#include <gmock/gmock.h>
#include <gtest/gtest.h>
using testing::An;
using testing::DoAll;
using testing::InSequence;
using testing::Return;
using testing::SetArgPointee;
using testing::StrictMock;
using testing::_;
namespace brillo {
class InputStreamSetTest : public testing::Test {
public:
void SetUp() override {
itf1_.reset(new StrictMock<MockStream>{});
itf2_.reset(new StrictMock<MockStream>{});
stream_.reset(new InputStreamSet({itf1_.get(), itf2_.get()}, {}, 100));
}
void TearDown() override {
stream_.reset();
itf2_.reset();
itf1_.reset();
}
std::unique_ptr<StrictMock<MockStream>> itf1_;
std::unique_ptr<StrictMock<MockStream>> itf2_;
std::unique_ptr<InputStreamSet> stream_;
inline static void* IntToPtr(int addr) {
return reinterpret_cast<void*>(addr);
}
};
TEST_F(InputStreamSetTest, InitialFalseAssumptions) {
// Methods that should just succeed/fail without calling underlying streams.
EXPECT_TRUE(stream_->CanRead());
EXPECT_FALSE(stream_->CanWrite());
EXPECT_FALSE(stream_->CanSeek());
EXPECT_EQ(100, stream_->GetSize());
EXPECT_FALSE(stream_->SetSizeBlocking(0, nullptr));
EXPECT_FALSE(stream_->GetPosition());
EXPECT_FALSE(stream_->Seek(0, Stream::Whence::FROM_BEGIN, nullptr, nullptr));
char buffer[100];
size_t size = 0;
EXPECT_FALSE(stream_->WriteAsync(buffer, sizeof(buffer), {}, {}, nullptr));
EXPECT_FALSE(stream_->WriteAllAsync(buffer, sizeof(buffer), {}, {}, nullptr));
EXPECT_FALSE(stream_->WriteNonBlocking(buffer, sizeof(buffer), &size,
nullptr));
EXPECT_FALSE(stream_->WriteBlocking(buffer, sizeof(buffer), &size, nullptr));
EXPECT_FALSE(stream_->WriteAllBlocking(buffer, sizeof(buffer), nullptr));
EXPECT_TRUE(stream_->FlushBlocking(nullptr));
EXPECT_TRUE(stream_->CloseBlocking(nullptr));
}
TEST_F(InputStreamSetTest, InitialTrueAssumptions) {
// Methods that redirect calls to underlying streams.
EXPECT_CALL(*itf1_, CanGetSize()).WillOnce(Return(true));
EXPECT_CALL(*itf2_, CanGetSize()).WillOnce(Return(true));
EXPECT_TRUE(stream_->CanGetSize());
// Reading from the first stream fails, so the second one shouldn't be used.
EXPECT_CALL(*itf1_, ReadNonBlocking(_, _, _, _, _))
.WillOnce(Return(false));
EXPECT_CALL(*itf2_, ReadNonBlocking(_, _, _, _, _)).Times(0);
char buffer[100];
size_t size = 0;
EXPECT_FALSE(stream_->ReadBlocking(buffer, sizeof(buffer), &size, nullptr));
}
TEST_F(InputStreamSetTest, CanGetSize) {
EXPECT_CALL(*itf1_, CanGetSize()).WillOnce(Return(true));
EXPECT_CALL(*itf2_, CanGetSize()).WillOnce(Return(true));
EXPECT_TRUE(stream_->CanGetSize());
EXPECT_CALL(*itf1_, CanGetSize()).WillOnce(Return(false));
EXPECT_FALSE(stream_->CanGetSize());
EXPECT_CALL(*itf1_, CanGetSize()).WillOnce(Return(true));
EXPECT_CALL(*itf2_, CanGetSize()).WillOnce(Return(false));
EXPECT_FALSE(stream_->CanGetSize());
}
TEST_F(InputStreamSetTest, GetRemainingSize) {
EXPECT_CALL(*itf1_, GetRemainingSize()).WillOnce(Return(10));
EXPECT_CALL(*itf2_, GetRemainingSize()).WillOnce(Return(32));
EXPECT_EQ(42, stream_->GetRemainingSize());
}
TEST_F(InputStreamSetTest, ReadNonBlocking) {
size_t read = 0;
bool eos = false;
InSequence s;
EXPECT_CALL(*itf1_, ReadNonBlocking(IntToPtr(1000), 100, _, _, _))
.WillOnce(DoAll(SetArgPointee<2>(10),
SetArgPointee<3>(false),
Return(true)));
EXPECT_TRUE(stream_->ReadNonBlocking(IntToPtr(1000), 100, &read, &eos,
nullptr));
EXPECT_EQ(10, read);
EXPECT_FALSE(eos);
EXPECT_CALL(*itf1_, ReadNonBlocking(IntToPtr(1000), 100, _, _, _))
.WillOnce(DoAll(SetArgPointee<2>(0), SetArgPointee<3>(true), Return(true)));
EXPECT_CALL(*itf2_, ReadNonBlocking(IntToPtr(1000), 100 , _, _, _))
.WillOnce(DoAll(SetArgPointee<2>(100),
SetArgPointee<3>(false),
Return(true)));
EXPECT_TRUE(stream_->ReadNonBlocking(IntToPtr(1000), 100, &read, &eos,
nullptr));
EXPECT_EQ(100, read);
EXPECT_FALSE(eos);
EXPECT_CALL(*itf2_, ReadNonBlocking(IntToPtr(1000), 100, _, _, _))
.WillOnce(DoAll(SetArgPointee<2>(0), SetArgPointee<3>(true), Return(true)));
EXPECT_TRUE(stream_->ReadNonBlocking(IntToPtr(1000), 100, &read, &eos,
nullptr));
EXPECT_EQ(0, read);
EXPECT_TRUE(eos);
}
TEST_F(InputStreamSetTest, ReadBlocking) {
size_t read = 0;
InSequence s;
EXPECT_CALL(*itf1_, ReadNonBlocking(IntToPtr(1000), 100, _, _, _))
.WillOnce(DoAll(SetArgPointee<2>(10),
SetArgPointee<3>(false),
Return(true)));
EXPECT_TRUE(stream_->ReadBlocking(IntToPtr(1000), 100, &read, nullptr));
EXPECT_EQ(10, read);
EXPECT_CALL(*itf1_, ReadNonBlocking(IntToPtr(1000), 100, _, _, _))
.WillOnce(DoAll(SetArgPointee<2>(0),
SetArgPointee<3>(true),
Return(true)));
EXPECT_CALL(*itf2_, ReadNonBlocking(IntToPtr(1000), 100, _, _, _))
.WillOnce(DoAll(SetArgPointee<2>(0),
SetArgPointee<3>(false),
Return(true)));
EXPECT_CALL(*itf2_, WaitForDataBlocking(Stream::AccessMode::READ, _, _, _))
.WillOnce(Return(true));
EXPECT_CALL(*itf2_, ReadNonBlocking(IntToPtr(1000), 100, _, _, _))
.WillOnce(DoAll(SetArgPointee<2>(100),
SetArgPointee<3>(false),
Return(true)));
EXPECT_TRUE(stream_->ReadBlocking(IntToPtr(1000), 100, &read, nullptr));
EXPECT_EQ(100, read);
EXPECT_CALL(*itf2_, ReadNonBlocking(IntToPtr(1000), 100, _, _, _))
.WillOnce(DoAll(SetArgPointee<2>(0),
SetArgPointee<3>(true),
Return(true)));
EXPECT_TRUE(stream_->ReadBlocking(IntToPtr(1000), 100, &read, nullptr));
EXPECT_EQ(0, read);
}
} // namespace brillo