523 lines
18 KiB
C++
523 lines
18 KiB
C++
// 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/stream.h>
|
|
|
|
#include <limits>
|
|
|
|
#include <base/callback.h>
|
|
#include <gmock/gmock.h>
|
|
#include <gtest/gtest.h>
|
|
|
|
#include <brillo/bind_lambda.h>
|
|
#include <brillo/message_loops/fake_message_loop.h>
|
|
#include <brillo/streams/stream_errors.h>
|
|
|
|
using testing::DoAll;
|
|
using testing::InSequence;
|
|
using testing::Return;
|
|
using testing::SaveArg;
|
|
using testing::SetArgPointee;
|
|
using testing::_;
|
|
|
|
namespace brillo {
|
|
|
|
using AccessMode = Stream::AccessMode;
|
|
using Whence = Stream::Whence;
|
|
|
|
// To verify "non-trivial" methods implemented in Stream, mock out the
|
|
// "trivial" methods to make sure the ones we are interested in testing
|
|
// actually end up calling the expected methods with right parameters.
|
|
class MockStreamImpl : public Stream {
|
|
public:
|
|
MockStreamImpl() = default;
|
|
|
|
MOCK_CONST_METHOD0(IsOpen, bool());
|
|
MOCK_CONST_METHOD0(CanRead, bool());
|
|
MOCK_CONST_METHOD0(CanWrite, bool());
|
|
MOCK_CONST_METHOD0(CanSeek, bool());
|
|
MOCK_CONST_METHOD0(CanGetSize, bool());
|
|
|
|
MOCK_CONST_METHOD0(GetSize, uint64_t());
|
|
MOCK_METHOD2(SetSizeBlocking, bool(uint64_t, ErrorPtr*));
|
|
MOCK_CONST_METHOD0(GetRemainingSize, uint64_t());
|
|
|
|
MOCK_CONST_METHOD0(GetPosition, uint64_t());
|
|
MOCK_METHOD4(Seek, bool(int64_t, Whence, uint64_t*, ErrorPtr*));
|
|
|
|
// Omitted: ReadAsync
|
|
// Omitted: ReadAllAsync
|
|
MOCK_METHOD5(ReadNonBlocking, bool(void*, size_t, size_t*, bool*, ErrorPtr*));
|
|
// Omitted: ReadBlocking
|
|
// Omitted: ReadAllBlocking
|
|
|
|
// Omitted: WriteAsync
|
|
// Omitted: WriteAllAsync
|
|
MOCK_METHOD4(WriteNonBlocking, bool(const void*, size_t, size_t*, ErrorPtr*));
|
|
// Omitted: WriteBlocking
|
|
// Omitted: WriteAllBlocking
|
|
|
|
MOCK_METHOD1(FlushBlocking, bool(ErrorPtr*));
|
|
MOCK_METHOD1(CloseBlocking, bool(ErrorPtr*));
|
|
|
|
MOCK_METHOD3(WaitForData, bool(AccessMode,
|
|
const base::Callback<void(AccessMode)>&,
|
|
ErrorPtr*));
|
|
MOCK_METHOD4(WaitForDataBlocking,
|
|
bool(AccessMode, base::TimeDelta, AccessMode*, ErrorPtr*));
|
|
|
|
private:
|
|
DISALLOW_COPY_AND_ASSIGN(MockStreamImpl);
|
|
};
|
|
|
|
TEST(Stream, TruncateBlocking) {
|
|
MockStreamImpl stream_mock;
|
|
EXPECT_CALL(stream_mock, GetPosition()).WillOnce(Return(123));
|
|
EXPECT_CALL(stream_mock, SetSizeBlocking(123, _)).WillOnce(Return(true));
|
|
EXPECT_TRUE(stream_mock.TruncateBlocking(nullptr));
|
|
}
|
|
|
|
TEST(Stream, SetPosition) {
|
|
MockStreamImpl stream_mock;
|
|
EXPECT_CALL(stream_mock, Seek(12345, Whence::FROM_BEGIN, _, _))
|
|
.WillOnce(Return(true));
|
|
EXPECT_TRUE(stream_mock.SetPosition(12345, nullptr));
|
|
|
|
// Test too large an offset (that doesn't fit in signed 64 bit value).
|
|
ErrorPtr error;
|
|
uint64_t max_offset = std::numeric_limits<int64_t>::max();
|
|
EXPECT_CALL(stream_mock, Seek(max_offset, _, _, _))
|
|
.WillOnce(Return(true));
|
|
EXPECT_TRUE(stream_mock.SetPosition(max_offset, nullptr));
|
|
|
|
EXPECT_FALSE(stream_mock.SetPosition(max_offset + 1, &error));
|
|
EXPECT_EQ(errors::stream::kDomain, error->GetDomain());
|
|
EXPECT_EQ(errors::stream::kInvalidParameter, error->GetCode());
|
|
}
|
|
|
|
TEST(Stream, ReadAsync) {
|
|
size_t read_size = 0;
|
|
bool succeeded = false;
|
|
bool failed = false;
|
|
auto success_callback = [](size_t* read_size, bool* succeeded,size_t size) {
|
|
*read_size = size;
|
|
*succeeded = true;
|
|
};
|
|
auto error_callback = [](bool* failed, const Error* /* error */) {
|
|
*failed = true;
|
|
};
|
|
|
|
MockStreamImpl stream_mock;
|
|
base::Callback<void(AccessMode)> data_callback;
|
|
char buf[10];
|
|
|
|
// This sets up an initial non blocking read that would block, so ReadAsync()
|
|
// should wait for more data.
|
|
EXPECT_CALL(stream_mock, ReadNonBlocking(buf, 10, _, _, _))
|
|
.WillOnce(
|
|
DoAll(SetArgPointee<2>(0), SetArgPointee<3>(false), Return(true)));
|
|
EXPECT_CALL(stream_mock, WaitForData(AccessMode::READ, _, _))
|
|
.WillOnce(DoAll(SaveArg<1>(&data_callback), Return(true)));
|
|
EXPECT_TRUE(stream_mock.ReadAsync(
|
|
buf,
|
|
sizeof(buf),
|
|
base::Bind(success_callback,
|
|
base::Unretained(&read_size),
|
|
base::Unretained(&succeeded)),
|
|
base::Bind(error_callback, base::Unretained(&failed)),
|
|
nullptr));
|
|
EXPECT_EQ(0u, read_size);
|
|
EXPECT_FALSE(succeeded);
|
|
EXPECT_FALSE(failed);
|
|
|
|
// Since the previous call is waiting for the data to be available, we can't
|
|
// schedule another read.
|
|
ErrorPtr error;
|
|
EXPECT_FALSE(stream_mock.ReadAsync(
|
|
buf,
|
|
sizeof(buf),
|
|
base::Bind(success_callback,
|
|
base::Unretained(&read_size),
|
|
base::Unretained(&succeeded)),
|
|
base::Bind(error_callback, base::Unretained(&failed)),
|
|
&error));
|
|
EXPECT_EQ(errors::stream::kDomain, error->GetDomain());
|
|
EXPECT_EQ(errors::stream::kOperationNotSupported, error->GetCode());
|
|
EXPECT_EQ("Another asynchronous operation is still pending",
|
|
error->GetMessage());
|
|
|
|
// Making the data available via data_callback should not schedule the
|
|
// success callback from the main loop and run it directly instead.
|
|
EXPECT_CALL(stream_mock, ReadNonBlocking(buf, 10, _, _, _))
|
|
.WillOnce(DoAll(SetArgPointee<2>(7),
|
|
SetArgPointee<3>(false),
|
|
Return(true)));
|
|
data_callback.Run(AccessMode::READ);
|
|
EXPECT_EQ(7u, read_size);
|
|
EXPECT_FALSE(failed);
|
|
}
|
|
|
|
TEST(Stream, ReadAsync_DontWaitForData) {
|
|
bool succeeded = false;
|
|
bool failed = false;
|
|
auto success_callback = [](bool* succeeded, size_t /* size */) {
|
|
*succeeded = true;
|
|
};
|
|
auto error_callback = [](bool* failed, const Error* /* error */) {
|
|
*failed = true;
|
|
};
|
|
|
|
MockStreamImpl stream_mock;
|
|
char buf[10];
|
|
FakeMessageLoop fake_loop_{nullptr};
|
|
fake_loop_.SetAsCurrent();
|
|
|
|
EXPECT_CALL(stream_mock, ReadNonBlocking(buf, 10, _, _, _))
|
|
.WillOnce(
|
|
DoAll(SetArgPointee<2>(5), SetArgPointee<3>(false), Return(true)));
|
|
EXPECT_CALL(stream_mock, WaitForData(_, _, _)).Times(0);
|
|
EXPECT_TRUE(stream_mock.ReadAsync(
|
|
buf,
|
|
sizeof(buf),
|
|
base::Bind(success_callback, base::Unretained(&succeeded)),
|
|
base::Bind(error_callback, base::Unretained(&failed)),
|
|
nullptr));
|
|
// Even if ReadNonBlocking() returned some data without waiting, the
|
|
// |success_callback| should not run yet.
|
|
EXPECT_TRUE(fake_loop_.PendingTasks());
|
|
EXPECT_FALSE(succeeded);
|
|
EXPECT_FALSE(failed);
|
|
|
|
// Since the previous callback is still waiting in the main loop, we can't
|
|
// schedule another read yet.
|
|
ErrorPtr error;
|
|
EXPECT_FALSE(stream_mock.ReadAsync(
|
|
buf,
|
|
sizeof(buf),
|
|
base::Bind(success_callback, base::Unretained(&succeeded)),
|
|
base::Bind(error_callback, base::Unretained(&failed)),
|
|
&error));
|
|
EXPECT_EQ(errors::stream::kDomain, error->GetDomain());
|
|
EXPECT_EQ(errors::stream::kOperationNotSupported, error->GetCode());
|
|
EXPECT_EQ("Another asynchronous operation is still pending",
|
|
error->GetMessage());
|
|
|
|
fake_loop_.Run();
|
|
EXPECT_TRUE(succeeded);
|
|
EXPECT_FALSE(failed);
|
|
}
|
|
|
|
TEST(Stream, ReadAllAsync) {
|
|
bool succeeded = false;
|
|
bool failed = false;
|
|
auto success_callback = [](bool* succeeded) { *succeeded = true; };
|
|
auto error_callback = [](bool* failed, const Error* /* error */) {
|
|
*failed = true;
|
|
};
|
|
|
|
MockStreamImpl stream_mock;
|
|
base::Callback<void(AccessMode)> data_callback;
|
|
char buf[10];
|
|
|
|
// This sets up an initial non blocking read that would block, so
|
|
// ReadAllAsync() should wait for more data.
|
|
EXPECT_CALL(stream_mock, ReadNonBlocking(buf, 10, _, _, _))
|
|
.WillOnce(
|
|
DoAll(SetArgPointee<2>(0), SetArgPointee<3>(false), Return(true)));
|
|
EXPECT_CALL(stream_mock, WaitForData(AccessMode::READ, _, _))
|
|
.WillOnce(DoAll(SaveArg<1>(&data_callback), Return(true)));
|
|
EXPECT_TRUE(stream_mock.ReadAllAsync(
|
|
buf,
|
|
sizeof(buf),
|
|
base::Bind(success_callback, base::Unretained(&succeeded)),
|
|
base::Bind(error_callback, base::Unretained(&failed)),
|
|
nullptr));
|
|
EXPECT_FALSE(succeeded);
|
|
EXPECT_FALSE(failed);
|
|
testing::Mock::VerifyAndClearExpectations(&stream_mock);
|
|
|
|
// ReadAllAsync() will try to read non blocking until the read would block
|
|
// before it waits for the data to be available again.
|
|
EXPECT_CALL(stream_mock, ReadNonBlocking(buf, 10, _, _, _))
|
|
.WillOnce(DoAll(SetArgPointee<2>(7),
|
|
SetArgPointee<3>(false),
|
|
Return(true)));
|
|
EXPECT_CALL(stream_mock, ReadNonBlocking(buf + 7, 3, _, _, _))
|
|
.WillOnce(
|
|
DoAll(SetArgPointee<2>(0), SetArgPointee<3>(false), Return(true)));
|
|
EXPECT_CALL(stream_mock, WaitForData(AccessMode::READ, _, _))
|
|
.WillOnce(DoAll(SaveArg<1>(&data_callback), Return(true)));
|
|
data_callback.Run(AccessMode::READ);
|
|
EXPECT_FALSE(succeeded);
|
|
EXPECT_FALSE(failed);
|
|
testing::Mock::VerifyAndClearExpectations(&stream_mock);
|
|
|
|
EXPECT_CALL(stream_mock, ReadNonBlocking(buf + 7, 3, _, _, _))
|
|
.WillOnce(DoAll(SetArgPointee<2>(3),
|
|
SetArgPointee<3>(true),
|
|
Return(true)));
|
|
data_callback.Run(AccessMode::READ);
|
|
EXPECT_TRUE(succeeded);
|
|
EXPECT_FALSE(failed);
|
|
}
|
|
|
|
TEST(Stream, ReadAllAsync_EOS) {
|
|
bool succeeded = false;
|
|
bool failed = false;
|
|
auto success_callback = [](bool* succeeded) { *succeeded = true; };
|
|
auto error_callback = [](bool* failed, const Error* error) {
|
|
ASSERT_EQ(errors::stream::kDomain, error->GetDomain());
|
|
ASSERT_EQ(errors::stream::kPartialData, error->GetCode());
|
|
*failed = true;
|
|
};
|
|
|
|
MockStreamImpl stream_mock;
|
|
base::Callback<void(AccessMode)> data_callback;
|
|
char buf[10];
|
|
|
|
EXPECT_CALL(stream_mock, ReadNonBlocking(buf, 10, _, _, _))
|
|
.WillOnce(
|
|
DoAll(SetArgPointee<2>(0), SetArgPointee<3>(false), Return(true)));
|
|
EXPECT_CALL(stream_mock, WaitForData(AccessMode::READ, _, _))
|
|
.WillOnce(DoAll(SaveArg<1>(&data_callback), Return(true)));
|
|
EXPECT_TRUE(stream_mock.ReadAllAsync(
|
|
buf,
|
|
sizeof(buf),
|
|
base::Bind(success_callback, base::Unretained(&succeeded)),
|
|
base::Bind(error_callback, base::Unretained(&failed)),
|
|
nullptr));
|
|
|
|
// ReadAsyncAll() should finish and fail once ReadNonBlocking() returns an
|
|
// end-of-stream condition.
|
|
EXPECT_CALL(stream_mock, ReadNonBlocking(buf, 10, _, _, _))
|
|
.WillOnce(DoAll(SetArgPointee<2>(7),
|
|
SetArgPointee<3>(true),
|
|
Return(true)));
|
|
data_callback.Run(AccessMode::READ);
|
|
EXPECT_FALSE(succeeded);
|
|
EXPECT_TRUE(failed);
|
|
}
|
|
|
|
TEST(Stream, ReadBlocking) {
|
|
MockStreamImpl stream_mock;
|
|
char buf[1024];
|
|
size_t read = 0;
|
|
|
|
EXPECT_CALL(stream_mock, ReadNonBlocking(buf, 1024, _, _, _))
|
|
.WillOnce(DoAll(SetArgPointee<2>(24),
|
|
SetArgPointee<3>(false),
|
|
Return(true)));
|
|
EXPECT_TRUE(stream_mock.ReadBlocking(buf, sizeof(buf), &read, nullptr));
|
|
EXPECT_EQ(24, read);
|
|
|
|
EXPECT_CALL(stream_mock, ReadNonBlocking(buf, 1024, _, _, _))
|
|
.WillOnce(DoAll(SetArgPointee<2>(0),
|
|
SetArgPointee<3>(true),
|
|
Return(true)));
|
|
EXPECT_TRUE(stream_mock.ReadBlocking(buf, sizeof(buf), &read, nullptr));
|
|
EXPECT_EQ(0, read);
|
|
|
|
{
|
|
InSequence seq;
|
|
EXPECT_CALL(stream_mock, ReadNonBlocking(buf, 1024, _, _, _))
|
|
.WillOnce(DoAll(SetArgPointee<2>(0),
|
|
SetArgPointee<3>(false),
|
|
Return(true)));
|
|
EXPECT_CALL(stream_mock, WaitForDataBlocking(AccessMode::READ, _, _, _))
|
|
.WillOnce(Return(true));
|
|
EXPECT_CALL(stream_mock, ReadNonBlocking(buf, 1024, _, _, _))
|
|
.WillOnce(DoAll(SetArgPointee<2>(0),
|
|
SetArgPointee<3>(false),
|
|
Return(true)));
|
|
EXPECT_CALL(stream_mock, WaitForDataBlocking(AccessMode::READ, _, _, _))
|
|
.WillOnce(Return(true));
|
|
EXPECT_CALL(stream_mock, ReadNonBlocking(buf, 1024, _, _, _))
|
|
.WillOnce(DoAll(SetArgPointee<2>(124),
|
|
SetArgPointee<3>(false),
|
|
Return(true)));
|
|
}
|
|
EXPECT_TRUE(stream_mock.ReadBlocking(buf, sizeof(buf), &read, nullptr));
|
|
EXPECT_EQ(124, read);
|
|
|
|
{
|
|
InSequence seq;
|
|
EXPECT_CALL(stream_mock, ReadNonBlocking(buf, 1024, _, _, _))
|
|
.WillOnce(DoAll(SetArgPointee<2>(0),
|
|
SetArgPointee<3>(false),
|
|
Return(true)));
|
|
EXPECT_CALL(stream_mock, WaitForDataBlocking(AccessMode::READ, _, _, _))
|
|
.WillOnce(Return(false));
|
|
}
|
|
EXPECT_FALSE(stream_mock.ReadBlocking(buf, sizeof(buf), &read, nullptr));
|
|
}
|
|
|
|
TEST(Stream, ReadAllBlocking) {
|
|
class MockReadBlocking : public MockStreamImpl {
|
|
public:
|
|
MOCK_METHOD4(ReadBlocking, bool(void*, size_t, size_t*, ErrorPtr*));
|
|
} stream_mock;
|
|
|
|
char buf[1024];
|
|
|
|
EXPECT_CALL(stream_mock, ReadBlocking(buf, 1024, _, _))
|
|
.WillOnce(DoAll(SetArgPointee<2>(24), Return(true)));
|
|
EXPECT_CALL(stream_mock, ReadBlocking(buf + 24, 1000, _, _))
|
|
.WillOnce(DoAll(SetArgPointee<2>(1000), Return(true)));
|
|
EXPECT_TRUE(stream_mock.ReadAllBlocking(buf, sizeof(buf), nullptr));
|
|
|
|
ErrorPtr error;
|
|
EXPECT_CALL(stream_mock, ReadBlocking(buf, 1024, _, _))
|
|
.WillOnce(DoAll(SetArgPointee<2>(24), Return(true)));
|
|
EXPECT_CALL(stream_mock, ReadBlocking(buf + 24, 1000, _, _))
|
|
.WillOnce(DoAll(SetArgPointee<2>(0), Return(true)));
|
|
EXPECT_FALSE(stream_mock.ReadAllBlocking(buf, sizeof(buf), &error));
|
|
EXPECT_EQ(errors::stream::kDomain, error->GetDomain());
|
|
EXPECT_EQ(errors::stream::kPartialData, error->GetCode());
|
|
}
|
|
|
|
TEST(Stream, WriteAsync) {
|
|
size_t write_size = 0;
|
|
bool failed = false;
|
|
auto success_callback = [](size_t* write_size, size_t size) {
|
|
*write_size = size;
|
|
};
|
|
auto error_callback = [](bool* failed, const Error* /* error */) {
|
|
*failed = true;
|
|
};
|
|
|
|
MockStreamImpl stream_mock;
|
|
InSequence s;
|
|
base::Callback<void(AccessMode)> data_callback;
|
|
char buf[10] = {};
|
|
|
|
// WriteNonBlocking returns a blocking situation (size_written = 0) so the
|
|
// WaitForData() is run.
|
|
EXPECT_CALL(stream_mock, WriteNonBlocking(buf, 10, _, _))
|
|
.WillOnce(DoAll(SetArgPointee<2>(0), Return(true)));
|
|
EXPECT_CALL(stream_mock, WaitForData(AccessMode::WRITE, _, _))
|
|
.WillOnce(DoAll(SaveArg<1>(&data_callback), Return(true)));
|
|
EXPECT_TRUE(stream_mock.WriteAsync(
|
|
buf,
|
|
sizeof(buf),
|
|
base::Bind(success_callback, base::Unretained(&write_size)),
|
|
base::Bind(error_callback, base::Unretained(&failed)),
|
|
nullptr));
|
|
EXPECT_EQ(0u, write_size);
|
|
EXPECT_FALSE(failed);
|
|
|
|
ErrorPtr error;
|
|
EXPECT_FALSE(stream_mock.WriteAsync(
|
|
buf,
|
|
sizeof(buf),
|
|
base::Bind(success_callback, base::Unretained(&write_size)),
|
|
base::Bind(error_callback, base::Unretained(&failed)),
|
|
&error));
|
|
EXPECT_EQ(errors::stream::kDomain, error->GetDomain());
|
|
EXPECT_EQ(errors::stream::kOperationNotSupported, error->GetCode());
|
|
EXPECT_EQ("Another asynchronous operation is still pending",
|
|
error->GetMessage());
|
|
|
|
EXPECT_CALL(stream_mock, WriteNonBlocking(buf, 10, _, _))
|
|
.WillOnce(DoAll(SetArgPointee<2>(7), Return(true)));
|
|
data_callback.Run(AccessMode::WRITE);
|
|
EXPECT_EQ(7u, write_size);
|
|
EXPECT_FALSE(failed);
|
|
}
|
|
|
|
TEST(Stream, WriteAllAsync) {
|
|
bool succeeded = false;
|
|
bool failed = false;
|
|
auto success_callback = [](bool* succeeded) { *succeeded = true; };
|
|
auto error_callback = [](bool* failed, const Error* /* error */) {
|
|
*failed = true;
|
|
};
|
|
|
|
MockStreamImpl stream_mock;
|
|
base::Callback<void(AccessMode)> data_callback;
|
|
char buf[10] = {};
|
|
|
|
EXPECT_CALL(stream_mock, WriteNonBlocking(buf, 10, _, _))
|
|
.WillOnce(DoAll(SetArgPointee<2>(0), Return(true)));
|
|
EXPECT_CALL(stream_mock, WaitForData(AccessMode::WRITE, _, _))
|
|
.WillOnce(DoAll(SaveArg<1>(&data_callback), Return(true)));
|
|
EXPECT_TRUE(stream_mock.WriteAllAsync(
|
|
buf,
|
|
sizeof(buf),
|
|
base::Bind(success_callback, base::Unretained(&succeeded)),
|
|
base::Bind(error_callback, base::Unretained(&failed)),
|
|
nullptr));
|
|
testing::Mock::VerifyAndClearExpectations(&stream_mock);
|
|
EXPECT_FALSE(succeeded);
|
|
EXPECT_FALSE(failed);
|
|
|
|
EXPECT_CALL(stream_mock, WriteNonBlocking(buf, 10, _, _))
|
|
.WillOnce(DoAll(SetArgPointee<2>(7), Return(true)));
|
|
EXPECT_CALL(stream_mock, WriteNonBlocking(buf + 7, 3, _, _))
|
|
.WillOnce(DoAll(SetArgPointee<2>(0), Return(true)));
|
|
EXPECT_CALL(stream_mock, WaitForData(AccessMode::WRITE, _, _))
|
|
.WillOnce(DoAll(SaveArg<1>(&data_callback), Return(true)));
|
|
data_callback.Run(AccessMode::WRITE);
|
|
testing::Mock::VerifyAndClearExpectations(&stream_mock);
|
|
EXPECT_FALSE(succeeded);
|
|
EXPECT_FALSE(failed);
|
|
|
|
EXPECT_CALL(stream_mock, WriteNonBlocking(buf + 7, 3, _, _))
|
|
.WillOnce(DoAll(SetArgPointee<2>(3), Return(true)));
|
|
data_callback.Run(AccessMode::WRITE);
|
|
EXPECT_TRUE(succeeded);
|
|
EXPECT_FALSE(failed);
|
|
}
|
|
|
|
TEST(Stream, WriteBlocking) {
|
|
MockStreamImpl stream_mock;
|
|
char buf[1024];
|
|
size_t written = 0;
|
|
|
|
EXPECT_CALL(stream_mock, WriteNonBlocking(buf, 1024, _, _))
|
|
.WillOnce(DoAll(SetArgPointee<2>(24), Return(true)));
|
|
EXPECT_TRUE(stream_mock.WriteBlocking(buf, sizeof(buf), &written, nullptr));
|
|
EXPECT_EQ(24, written);
|
|
|
|
{
|
|
InSequence seq;
|
|
EXPECT_CALL(stream_mock, WriteNonBlocking(buf, 1024, _, _))
|
|
.WillOnce(DoAll(SetArgPointee<2>(0), Return(true)));
|
|
EXPECT_CALL(stream_mock, WaitForDataBlocking(AccessMode::WRITE, _, _, _))
|
|
.WillOnce(Return(true));
|
|
EXPECT_CALL(stream_mock, WriteNonBlocking(buf, 1024, _, _))
|
|
.WillOnce(DoAll(SetArgPointee<2>(0), Return(true)));
|
|
EXPECT_CALL(stream_mock, WaitForDataBlocking(AccessMode::WRITE, _, _, _))
|
|
.WillOnce(Return(true));
|
|
EXPECT_CALL(stream_mock, WriteNonBlocking(buf, 1024, _, _))
|
|
.WillOnce(DoAll(SetArgPointee<2>(124), Return(true)));
|
|
}
|
|
EXPECT_TRUE(stream_mock.WriteBlocking(buf, sizeof(buf), &written, nullptr));
|
|
EXPECT_EQ(124, written);
|
|
|
|
{
|
|
InSequence seq;
|
|
EXPECT_CALL(stream_mock, WriteNonBlocking(buf, 1024, _, _))
|
|
.WillOnce(DoAll(SetArgPointee<2>(0), Return(true)));
|
|
EXPECT_CALL(stream_mock, WaitForDataBlocking(AccessMode::WRITE, _, _, _))
|
|
.WillOnce(Return(false));
|
|
}
|
|
EXPECT_FALSE(stream_mock.WriteBlocking(buf, sizeof(buf), &written, nullptr));
|
|
}
|
|
|
|
TEST(Stream, WriteAllBlocking) {
|
|
class MockWritelocking : public MockStreamImpl {
|
|
public:
|
|
MOCK_METHOD4(WriteBlocking, bool(const void*, size_t, size_t*, ErrorPtr*));
|
|
} stream_mock;
|
|
|
|
char buf[1024];
|
|
|
|
EXPECT_CALL(stream_mock, WriteBlocking(buf, 1024, _, _))
|
|
.WillOnce(DoAll(SetArgPointee<2>(24), Return(true)));
|
|
EXPECT_CALL(stream_mock, WriteBlocking(buf + 24, 1000, _, _))
|
|
.WillOnce(DoAll(SetArgPointee<2>(1000), Return(true)));
|
|
EXPECT_TRUE(stream_mock.WriteAllBlocking(buf, sizeof(buf), nullptr));
|
|
}
|
|
|
|
} // namespace brillo
|