#include <pdx/service.h>
#include <memory>
#include <string>
#include <gmock/gmock.h>
#include <pdx/mock_service_endpoint.h>
using android::pdx::BorrowedChannelHandle;
using android::pdx::BorrowedHandle;
using android::pdx::Channel;
using android::pdx::ChannelReference;
using android::pdx::ErrorStatus;
using android::pdx::FileReference;
using android::pdx::LocalChannelHandle;
using android::pdx::LocalHandle;
using android::pdx::Message;
using android::pdx::MessageInfo;
using android::pdx::MockEndpoint;
using android::pdx::RemoteChannelHandle;
using android::pdx::RemoteHandle;
using android::pdx::Service;
using android::pdx::Status;
using testing::A;
using testing::ByMove;
using testing::DoAll;
using testing::Invoke;
using testing::Matcher;
using testing::Ref;
using testing::Return;
using testing::SetArgPointee;
using testing::WithArg;
using testing::WithoutArgs;
using testing::_;
namespace {
// Helper functions to construct fake void pointers for tests.
inline void* IntToPtr(intptr_t addr) { return reinterpret_cast<void*>(addr); }
// Helper matchers for working with iovec structures in tests.
// Simple matcher for one element iovec array:
// EXPECT_CALL(mock, method(IoVecMatcher(ptr, size)));
MATCHER_P2(IoVecMatcher, ptr, size, "") {
return arg->iov_base == ptr && arg->iov_len == size;
}
// Matcher for an array of iovecs:
// EXPECT_CALL(mock,
// method(IoVecMatcher(IoVecArray{{ptr1, size1}, {ptr2, size2}})));
using IoVecArray = std::vector<iovec>;
MATCHER_P(IoVecMatcher, iovec_array, "") {
for (const iovec& item : iovec_array) {
if (arg->iov_base != item.iov_base || arg->iov_len != item.iov_len)
return false;
arg++;
}
return true;
}
using IoVecData = std::vector<std::string>;
MATCHER_P(IoVecDataMatcher, iovec_data, "") {
for (const std::string& item : iovec_data) {
std::string data{reinterpret_cast<const char*>(arg->iov_base),
arg->iov_len};
if (data != item)
return false;
arg++;
}
return true;
}
MATCHER_P(FileHandleMatcher, value, "") { return arg.Get() == value; }
MATCHER_P(ChannelHandleMatcher, value, "") { return arg.value() == value; }
enum : int {
kTestPid = 1,
kTestTid,
kTestCid,
kTestMid,
kTestEuid,
kTestEgid,
kTestOp,
};
class MockService : public Service {
public:
using Service::Service;
MOCK_METHOD1(OnChannelOpen, std::shared_ptr<Channel>(Message& message));
MOCK_METHOD2(OnChannelClose,
void(Message& message, const std::shared_ptr<Channel>& channel));
MOCK_METHOD1(HandleMessage, Status<void>(Message& message));
MOCK_METHOD1(HandleImpulse, void(Message& impulse));
MOCK_METHOD0(OnSysPropChange, void());
MOCK_METHOD1(DumpState, std::string(size_t max_length));
};
class ServiceTest : public testing::Test {
public:
ServiceTest() {
auto endpoint = std::make_unique<testing::StrictMock<MockEndpoint>>();
EXPECT_CALL(*endpoint, SetService(_))
.Times(2)
.WillRepeatedly(Return(Status<void>{}));
service_ = std::make_shared<MockService>("MockSvc", std::move(endpoint));
}
MockEndpoint* endpoint() {
return static_cast<MockEndpoint*>(service_->endpoint());
}
void SetupMessageInfo(MessageInfo* info, int32_t op, bool impulse = false) {
info->pid = kTestPid;
info->tid = kTestTid;
info->cid = kTestCid;
info->mid = impulse ? Message::IMPULSE_MESSAGE_ID : kTestMid;
info->euid = kTestEuid;
info->egid = kTestEgid;
info->op = op;
info->flags = 0;
info->service = service_.get();
info->channel = nullptr;
info->send_len = 0;
info->recv_len = 0;
info->fd_count = 0;
memset(info->impulse, 0, sizeof(info->impulse));
}
void SetupMessageInfoAndDefaultExpectations(MessageInfo* info, int32_t op,
bool impulse = false) {
SetupMessageInfo(info, op, impulse);
EXPECT_CALL(*endpoint(), AllocateMessageState()).WillOnce(Return(kState));
EXPECT_CALL(*endpoint(), FreeMessageState(kState));
}
void ExpectDefaultHandleMessage() {
EXPECT_CALL(*endpoint(), MessageReply(_, -EOPNOTSUPP))
.WillOnce(Return(Status<void>{}));
}
std::shared_ptr<MockService> service_;
void* kState = IntToPtr(123456);
};
class ServiceMessageTest : public ServiceTest {
public:
ServiceMessageTest() {
MessageInfo info;
SetupMessageInfoAndDefaultExpectations(&info, kTestOp);
message_ = std::make_unique<Message>(info);
}
std::unique_ptr<Message> message_;
};
} // anonymous namespace
///////////////////////////////////////////////////////////////////////////////
// Service class tests
///////////////////////////////////////////////////////////////////////////////
TEST_F(ServiceTest, IsInitialized) {
EXPECT_TRUE(service_->IsInitialized());
service_ = std::make_shared<MockService>("MockSvc2", nullptr);
EXPECT_FALSE(service_->IsInitialized());
}
TEST_F(ServiceTest, ConstructMessage) {
MessageInfo info;
SetupMessageInfo(&info, kTestOp);
auto test_channel = std::make_shared<Channel>();
info.channel = test_channel.get();
EXPECT_CALL(*endpoint(), AllocateMessageState()).WillOnce(Return(kState));
Message message{info};
EXPECT_FALSE(message.IsImpulse());
EXPECT_EQ(kTestPid, message.GetProcessId());
EXPECT_EQ(kTestTid, message.GetThreadId());
EXPECT_EQ(kTestCid, message.GetChannelId());
EXPECT_EQ(kTestMid, message.GetMessageId());
EXPECT_EQ((unsigned) kTestEuid, message.GetEffectiveUserId());
EXPECT_EQ((unsigned) kTestEgid, message.GetEffectiveGroupId());
EXPECT_EQ(kTestOp, message.GetOp());
EXPECT_EQ(service_, message.GetService());
EXPECT_EQ(test_channel, message.GetChannel());
EXPECT_FALSE(message.replied());
EXPECT_FALSE(message.IsChannelExpired());
EXPECT_FALSE(message.IsServiceExpired());
EXPECT_EQ(kState, message.GetState());
ExpectDefaultHandleMessage();
EXPECT_CALL(*endpoint(), FreeMessageState(kState));
}
TEST_F(ServiceTest, ConstructImpulseMessage) {
MessageInfo info;
SetupMessageInfo(&info, kTestOp, true);
auto test_channel = std::make_shared<Channel>();
info.channel = test_channel.get();
EXPECT_CALL(*endpoint(), AllocateMessageState()).WillOnce(Return(nullptr));
Message message{info};
EXPECT_TRUE(message.IsImpulse());
EXPECT_EQ(kTestOp, message.GetOp());
EXPECT_EQ(service_, message.GetService());
EXPECT_EQ(test_channel, message.GetChannel());
EXPECT_TRUE(message.replied());
EXPECT_FALSE(message.IsChannelExpired());
EXPECT_FALSE(message.IsServiceExpired());
// DefaultHandleMessage should not be called here.
EXPECT_CALL(*endpoint(), FreeMessageState(nullptr));
}
TEST_F(ServiceTest, HandleMessageChannelOpen) {
MessageInfo info;
SetupMessageInfoAndDefaultExpectations(&info,
android::pdx::opcodes::CHANNEL_OPEN);
Message message{info};
auto channel = std::make_shared<Channel>();
EXPECT_CALL(*service_, OnChannelOpen(Ref(message))).WillOnce(Return(channel));
EXPECT_CALL(*endpoint(), SetChannel(kTestCid, channel.get()))
.WillOnce(Return(Status<void>{}));
EXPECT_CALL(*endpoint(), MessageReply(&message, 0))
.WillOnce(Return(Status<void>{}));
EXPECT_TRUE(service_->Service::HandleMessage(message));
}
TEST_F(ServiceTest, HandleMessageChannelClose) {
MessageInfo info;
SetupMessageInfoAndDefaultExpectations(&info,
android::pdx::opcodes::CHANNEL_CLOSE);
auto channel = std::make_shared<Channel>();
info.channel = channel.get();
Message message{info};
EXPECT_CALL(*service_, OnChannelClose(Ref(message), channel));
EXPECT_CALL(*endpoint(), SetChannel(kTestCid, nullptr))
.WillOnce(Return(Status<void>{}));
EXPECT_CALL(*endpoint(), MessageReply(&message, 0))
.WillOnce(Return(Status<void>{}));
EXPECT_TRUE(service_->Service::HandleMessage(message));
}
TEST_F(ServiceTest, HandleMessageOnSysPropChange) {
MessageInfo info;
SetupMessageInfoAndDefaultExpectations(
&info, android::pdx::opcodes::REPORT_SYSPROP_CHANGE);
Message message{info};
EXPECT_CALL(*service_, OnSysPropChange());
EXPECT_CALL(*endpoint(), MessageReply(&message, 0))
.WillOnce(Return(Status<void>{}));
EXPECT_TRUE(service_->Service::HandleMessage(message));
}
TEST_F(ServiceTest, HandleMessageOnDumpState) {
const size_t kRecvBufSize = 1000;
MessageInfo info;
SetupMessageInfoAndDefaultExpectations(&info,
android::pdx::opcodes::DUMP_STATE);
info.recv_len = kRecvBufSize;
Message message{info};
const std::string kReply = "foo";
EXPECT_CALL(*service_, DumpState(kRecvBufSize)).WillOnce(Return(kReply));
EXPECT_CALL(
*endpoint(),
WriteMessageData(&message, IoVecDataMatcher(IoVecData{kReply}), 1))
.WillOnce(Return(kReply.size()));
EXPECT_CALL(*endpoint(), MessageReply(&message, kReply.size()))
.WillOnce(Return(Status<void>{}));
EXPECT_TRUE(service_->Service::HandleMessage(message));
}
TEST_F(ServiceTest, HandleMessageOnDumpStateTooLarge) {
const size_t kRecvBufSize = 3;
MessageInfo info;
SetupMessageInfoAndDefaultExpectations(&info,
android::pdx::opcodes::DUMP_STATE);
info.recv_len = kRecvBufSize;
Message message{info};
const std::string kReply = "0123456789";
const std::string kActualReply = kReply.substr(0, kRecvBufSize);
EXPECT_CALL(*service_, DumpState(kRecvBufSize)).WillOnce(Return(kReply));
EXPECT_CALL(
*endpoint(),
WriteMessageData(&message, IoVecDataMatcher(IoVecData{kActualReply}), 1))
.WillOnce(Return(kActualReply.size()));
EXPECT_CALL(*endpoint(), MessageReply(&message, kActualReply.size()))
.WillOnce(Return(Status<void>{}));
EXPECT_TRUE(service_->Service::HandleMessage(message));
}
TEST_F(ServiceTest, HandleMessageOnDumpStateFail) {
const size_t kRecvBufSize = 1000;
MessageInfo info;
SetupMessageInfoAndDefaultExpectations(&info,
android::pdx::opcodes::DUMP_STATE);
info.recv_len = kRecvBufSize;
Message message{info};
const std::string kReply = "foo";
EXPECT_CALL(*service_, DumpState(kRecvBufSize)).WillOnce(Return(kReply));
EXPECT_CALL(
*endpoint(),
WriteMessageData(&message, IoVecDataMatcher(IoVecData{kReply}), 1))
.WillOnce(Return(1));
EXPECT_CALL(*endpoint(), MessageReply(&message, -EIO))
.WillOnce(Return(Status<void>{}));
EXPECT_TRUE(service_->Service::HandleMessage(message));
}
TEST_F(ServiceTest, HandleMessageCustom) {
MessageInfo info;
SetupMessageInfoAndDefaultExpectations(&info, kTestOp);
Message message{info};
EXPECT_CALL(*endpoint(), MessageReply(&message, -EOPNOTSUPP))
.WillOnce(Return(Status<void>{}));
EXPECT_TRUE(service_->Service::HandleMessage(message));
}
TEST_F(ServiceTest, ReplyMessageWithoutService) {
MessageInfo info;
SetupMessageInfo(&info, kTestOp);
EXPECT_CALL(*endpoint(), AllocateMessageState()).WillOnce(Return(nullptr));
Message message{info};
EXPECT_FALSE(message.IsServiceExpired());
service_.reset();
EXPECT_TRUE(message.IsServiceExpired());
EXPECT_EQ(EINVAL, message.Reply(12).error());
}
TEST_F(ServiceTest, ReceiveAndDispatchMessage) {
MessageInfo info;
SetupMessageInfoAndDefaultExpectations(&info, kTestOp);
ExpectDefaultHandleMessage();
auto on_receive = [&info](Message* message) -> Status<void> {
*message = Message{info};
return {};
};
EXPECT_CALL(*endpoint(), MessageReceive(_)).WillOnce(Invoke(on_receive));
EXPECT_CALL(*service_, HandleMessage(_)).WillOnce(Return(Status<void>{}));
EXPECT_TRUE(service_->ReceiveAndDispatch());
}
TEST_F(ServiceTest, ReceiveAndDispatchImpulse) {
MessageInfo info;
SetupMessageInfoAndDefaultExpectations(&info, kTestOp, true);
auto on_receive = [&info](Message* message) -> Status<void> {
*message = Message{info};
return {};
};
EXPECT_CALL(*endpoint(), MessageReceive(_)).WillOnce(Invoke(on_receive));
EXPECT_CALL(*service_, HandleImpulse(_));
EXPECT_TRUE(service_->ReceiveAndDispatch());
}
TEST_F(ServiceTest, Cancel) {
EXPECT_CALL(*endpoint(), Cancel()).WillOnce(Return(Status<void>{}));
EXPECT_TRUE(service_->Cancel());
}
///////////////////////////////////////////////////////////////////////////////
// Message class tests
///////////////////////////////////////////////////////////////////////////////
TEST_F(ServiceMessageTest, Reply) {
EXPECT_CALL(*endpoint(), MessageReply(message_.get(), 12))
.WillOnce(Return(Status<void>{}));
EXPECT_FALSE(message_->replied());
EXPECT_TRUE(message_->Reply(12));
EXPECT_TRUE(message_->replied());
EXPECT_EQ(EINVAL, message_->Reply(12).error()); // Already replied.
}
TEST_F(ServiceMessageTest, ReplyFail) {
EXPECT_CALL(*endpoint(), MessageReply(message_.get(), 12))
.WillOnce(Return(ErrorStatus{EIO}));
EXPECT_EQ(EIO, message_->Reply(12).error());
ExpectDefaultHandleMessage();
}
TEST_F(ServiceMessageTest, ReplyError) {
EXPECT_CALL(*endpoint(), MessageReply(message_.get(), -12))
.WillOnce(Return(Status<void>{}));
EXPECT_TRUE(message_->ReplyError(12));
}
TEST_F(ServiceMessageTest, ReplyFileDescriptor) {
EXPECT_CALL(*endpoint(), MessageReplyFd(message_.get(), 5))
.WillOnce(Return(Status<void>{}));
EXPECT_TRUE(message_->ReplyFileDescriptor(5));
}
TEST_F(ServiceMessageTest, ReplyLocalFileHandle) {
const int kFakeFd = 12345;
LocalHandle handle{kFakeFd};
EXPECT_CALL(*endpoint(), MessageReplyFd(message_.get(), kFakeFd))
.WillOnce(Return(Status<void>{}));
EXPECT_TRUE(message_->Reply(handle));
handle.Release(); // Make sure we do not close the fake file descriptor.
}
TEST_F(ServiceMessageTest, ReplyLocalFileHandleError) {
LocalHandle handle{-EINVAL};
EXPECT_CALL(*endpoint(), MessageReply(message_.get(), -EINVAL))
.WillOnce(Return(Status<void>{}));
EXPECT_TRUE(message_->Reply(handle));
}
TEST_F(ServiceMessageTest, ReplyBorrowedFileHandle) {
const int kFakeFd = 12345;
BorrowedHandle handle{kFakeFd};
EXPECT_CALL(*endpoint(), MessageReplyFd(message_.get(), kFakeFd))
.WillOnce(Return(Status<void>{}));
EXPECT_TRUE(message_->Reply(handle));
}
TEST_F(ServiceMessageTest, ReplyBorrowedFileHandleError) {
BorrowedHandle handle{-EACCES};
EXPECT_CALL(*endpoint(), MessageReply(message_.get(), -EACCES))
.WillOnce(Return(Status<void>{}));
EXPECT_TRUE(message_->Reply(handle));
}
TEST_F(ServiceMessageTest, ReplyRemoteFileHandle) {
RemoteHandle handle{123};
EXPECT_CALL(*endpoint(), MessageReply(message_.get(), handle.Get()))
.WillOnce(Return(Status<void>{}));
EXPECT_TRUE(message_->Reply(handle));
}
TEST_F(ServiceMessageTest, ReplyRemoteFileHandleError) {
RemoteHandle handle{-EIO};
EXPECT_CALL(*endpoint(), MessageReply(message_.get(), -EIO))
.WillOnce(Return(Status<void>{}));
EXPECT_TRUE(message_->Reply(handle));
}
TEST_F(ServiceMessageTest, ReplyLocalChannelHandle) {
LocalChannelHandle handle{nullptr, 12345};
EXPECT_CALL(*endpoint(), MessageReplyChannelHandle(
message_.get(), A<const LocalChannelHandle&>()))
.WillOnce(Return(Status<void>{}));
EXPECT_TRUE(message_->Reply(handle));
}
TEST_F(ServiceMessageTest, ReplyBorrowedChannelHandle) {
BorrowedChannelHandle handle{12345};
EXPECT_CALL(*endpoint(),
MessageReplyChannelHandle(message_.get(),
A<const BorrowedChannelHandle&>()))
.WillOnce(Return(Status<void>{}));
EXPECT_TRUE(message_->Reply(handle));
}
TEST_F(ServiceMessageTest, ReplyRemoteChannelHandle) {
RemoteChannelHandle handle{12345};
EXPECT_CALL(*endpoint(), MessageReplyChannelHandle(
message_.get(), A<const RemoteChannelHandle&>()))
.WillOnce(Return(Status<void>{}));
EXPECT_TRUE(message_->Reply(handle));
}
TEST_F(ServiceMessageTest, ReplyStatusInt) {
Status<int> status{123};
EXPECT_CALL(*endpoint(), MessageReply(message_.get(), status.get()))
.WillOnce(Return(Status<void>{}));
EXPECT_TRUE(message_->Reply(status));
}
TEST_F(ServiceMessageTest, ReplyStatusError) {
Status<int> status{ErrorStatus{EIO}};
EXPECT_CALL(*endpoint(), MessageReply(message_.get(), -status.error()))
.WillOnce(Return(Status<void>{}));
EXPECT_TRUE(message_->Reply(status));
}
TEST_F(ServiceMessageTest, Read) {
ExpectDefaultHandleMessage();
void* const kDataBuffer = IntToPtr(12345);
const size_t kDataSize = 100;
EXPECT_CALL(
*endpoint(),
ReadMessageData(message_.get(), IoVecMatcher(kDataBuffer, kDataSize), 1))
.WillOnce(Return(50))
.WillOnce(Return(ErrorStatus{EACCES}));
EXPECT_EQ(50u, message_->Read(kDataBuffer, kDataSize).get());
EXPECT_EQ(EACCES, message_->Read(kDataBuffer, kDataSize).error());
}
TEST_F(ServiceMessageTest, ReadVector) {
ExpectDefaultHandleMessage();
char buffer1[10];
char buffer2[20];
iovec vec[] = {{buffer1, sizeof(buffer1)}, {buffer2, sizeof(buffer2)}};
EXPECT_CALL(*endpoint(),
ReadMessageData(
message_.get(),
IoVecMatcher(IoVecArray{std::begin(vec), std::end(vec)}), 2))
.WillOnce(Return(30))
.WillOnce(Return(15))
.WillOnce(Return(ErrorStatus{EBADF}));
EXPECT_EQ(30u, message_->ReadVector(vec, 2).get());
EXPECT_EQ(15u, message_->ReadVector(vec).get());
EXPECT_EQ(EBADF, message_->ReadVector(vec).error());
}
TEST_F(ServiceMessageTest, Write) {
ExpectDefaultHandleMessage();
void* const kDataBuffer = IntToPtr(12345);
const size_t kDataSize = 100;
EXPECT_CALL(
*endpoint(),
WriteMessageData(message_.get(), IoVecMatcher(kDataBuffer, kDataSize), 1))
.WillOnce(Return(50))
.WillOnce(Return(ErrorStatus{EBADMSG}));
EXPECT_EQ(50u, message_->Write(kDataBuffer, kDataSize).get());
EXPECT_EQ(EBADMSG, message_->Write(kDataBuffer, kDataSize).error());
}
TEST_F(ServiceMessageTest, WriteVector) {
ExpectDefaultHandleMessage();
char buffer1[10];
char buffer2[20];
iovec vec[] = {{buffer1, sizeof(buffer1)}, {buffer2, sizeof(buffer2)}};
EXPECT_CALL(*endpoint(),
WriteMessageData(
message_.get(),
IoVecMatcher(IoVecArray{std::begin(vec), std::end(vec)}), 2))
.WillOnce(Return(30))
.WillOnce(Return(15))
.WillOnce(Return(ErrorStatus{EIO}));
EXPECT_EQ(30u, message_->WriteVector(vec, 2).get());
EXPECT_EQ(15u, message_->WriteVector(vec).get());
EXPECT_EQ(EIO, message_->WriteVector(vec, 2).error());
}
TEST_F(ServiceMessageTest, PushLocalFileHandle) {
ExpectDefaultHandleMessage();
const int kFakeFd = 12345;
LocalHandle handle{kFakeFd};
EXPECT_CALL(*endpoint(),
PushFileHandle(message_.get(), Matcher<const LocalHandle&>(
FileHandleMatcher(kFakeFd))))
.WillOnce(Return(12))
.WillOnce(Return(ErrorStatus{EIO}));
EXPECT_EQ(12, message_->PushFileHandle(handle).get());
EXPECT_EQ(EIO, message_->PushFileHandle(handle).error());
handle.Release(); // Make sure we do not close the fake file descriptor.
}
TEST_F(ServiceMessageTest, PushBorrowedFileHandle) {
ExpectDefaultHandleMessage();
const int kFakeFd = 12345;
BorrowedHandle handle{kFakeFd};
EXPECT_CALL(*endpoint(),
PushFileHandle(message_.get(), Matcher<const BorrowedHandle&>(
FileHandleMatcher(kFakeFd))))
.WillOnce(Return(13))
.WillOnce(Return(ErrorStatus{EACCES}));
EXPECT_EQ(13, message_->PushFileHandle(handle).get());
EXPECT_EQ(EACCES, message_->PushFileHandle(handle).error());
}
TEST_F(ServiceMessageTest, PushRemoteFileHandle) {
ExpectDefaultHandleMessage();
const int kFakeFd = 12345;
RemoteHandle handle{kFakeFd};
EXPECT_CALL(*endpoint(),
PushFileHandle(message_.get(), Matcher<const RemoteHandle&>(
FileHandleMatcher(kFakeFd))))
.WillOnce(Return(kFakeFd))
.WillOnce(Return(ErrorStatus{EIO}));
EXPECT_EQ(kFakeFd, message_->PushFileHandle(handle).get());
EXPECT_EQ(EIO, message_->PushFileHandle(handle).error());
}
TEST_F(ServiceMessageTest, PushLocalChannelHandle) {
ExpectDefaultHandleMessage();
int32_t kValue = 12345;
LocalChannelHandle handle{nullptr, kValue};
EXPECT_CALL(*endpoint(), PushChannelHandle(message_.get(),
Matcher<const LocalChannelHandle&>(
ChannelHandleMatcher(kValue))))
.WillOnce(Return(7))
.WillOnce(Return(ErrorStatus{EIO}));
EXPECT_EQ(7, message_->PushChannelHandle(handle).get());
EXPECT_EQ(EIO, message_->PushChannelHandle(handle).error());
}
TEST_F(ServiceMessageTest, PushBorrowedChannelHandle) {
ExpectDefaultHandleMessage();
int32_t kValue = 12345;
BorrowedChannelHandle handle{kValue};
EXPECT_CALL(
*endpoint(),
PushChannelHandle(message_.get(), Matcher<const BorrowedChannelHandle&>(
ChannelHandleMatcher(kValue))))
.WillOnce(Return(8))
.WillOnce(Return(ErrorStatus{EIO}));
EXPECT_EQ(8, message_->PushChannelHandle(handle).get());
EXPECT_EQ(EIO, message_->PushChannelHandle(handle).error());
}
TEST_F(ServiceMessageTest, PushRemoteChannelHandle) {
ExpectDefaultHandleMessage();
int32_t kValue = 12345;
RemoteChannelHandle handle{kValue};
EXPECT_CALL(
*endpoint(),
PushChannelHandle(message_.get(), Matcher<const RemoteChannelHandle&>(
ChannelHandleMatcher(kValue))))
.WillOnce(Return(kValue))
.WillOnce(Return(ErrorStatus{EIO}));
EXPECT_EQ(kValue, message_->PushChannelHandle(handle).get());
EXPECT_EQ(EIO, message_->PushChannelHandle(handle).error());
}
TEST_F(ServiceMessageTest, GetFileHandle) {
ExpectDefaultHandleMessage();
auto make_file_handle = [](FileReference ref) { return LocalHandle{ref}; };
EXPECT_CALL(*endpoint(), GetFileHandle(message_.get(), _))
.WillOnce(WithArg<1>(Invoke(make_file_handle)));
LocalHandle handle;
FileReference kRef = 12345;
EXPECT_TRUE(message_->GetFileHandle(kRef, &handle));
EXPECT_EQ(kRef, handle.Get());
handle.Release(); // Make sure we do not close the fake file descriptor.
}
TEST_F(ServiceMessageTest, GetFileHandleInvalid) {
ExpectDefaultHandleMessage();
LocalHandle handle;
FileReference kRef = -12;
EXPECT_TRUE(message_->GetFileHandle(kRef, &handle));
EXPECT_EQ(kRef, handle.Get());
}
TEST_F(ServiceMessageTest, GetFileHandleError) {
ExpectDefaultHandleMessage();
EXPECT_CALL(*endpoint(), GetFileHandle(message_.get(), _))
.WillOnce(WithoutArgs(Invoke([] { return LocalHandle{-EIO}; })));
LocalHandle handle;
FileReference kRef = 12345;
EXPECT_FALSE(message_->GetFileHandle(kRef, &handle));
EXPECT_EQ(-EIO, handle.Get());
}
TEST_F(ServiceMessageTest, GetChannelHandle) {
ExpectDefaultHandleMessage();
auto make_channel_handle = [](ChannelReference ref) {
return LocalChannelHandle{nullptr, ref};
};
EXPECT_CALL(*endpoint(), GetChannelHandle(message_.get(), _))
.WillOnce(WithArg<1>(Invoke(make_channel_handle)));
LocalChannelHandle handle;
ChannelReference kRef = 12345;
EXPECT_TRUE(message_->GetChannelHandle(kRef, &handle));
EXPECT_EQ(kRef, handle.value());
}
TEST_F(ServiceMessageTest, GetChannelHandleInvalid) {
ExpectDefaultHandleMessage();
LocalChannelHandle handle;
ChannelReference kRef = -12;
EXPECT_TRUE(message_->GetChannelHandle(kRef, &handle));
EXPECT_EQ(-12, handle.value());
}
TEST_F(ServiceMessageTest, GetChannelHandleError) {
ExpectDefaultHandleMessage();
EXPECT_CALL(*endpoint(), GetChannelHandle(message_.get(), _))
.WillOnce(WithoutArgs(Invoke([] {
return LocalChannelHandle{nullptr, -EIO};
})));
LocalChannelHandle handle;
ChannelReference kRef = 12345;
EXPECT_FALSE(message_->GetChannelHandle(kRef, &handle));
EXPECT_EQ(-EIO, handle.value());
}
TEST_F(ServiceMessageTest, ModifyChannelEvents) {
ExpectDefaultHandleMessage();
int kClearMask = 1;
int kSetMask = 2;
EXPECT_CALL(*endpoint(), ModifyChannelEvents(kTestCid, kClearMask, kSetMask))
.WillOnce(Return(Status<void>{}));
EXPECT_TRUE(message_->ModifyChannelEvents(kClearMask, kSetMask));
}
TEST_F(ServiceMessageTest, PushChannelSameService) {
ExpectDefaultHandleMessage();
int kFlags = 123;
int32_t kValue = 12;
EXPECT_CALL(*endpoint(), PushChannel(message_.get(), kFlags, nullptr, _))
.WillOnce(DoAll(SetArgPointee<3>(kTestCid),
Return(ByMove(RemoteChannelHandle{kValue}))));
int channel_id = -1;
auto status = message_->PushChannel(kFlags, nullptr, &channel_id);
ASSERT_TRUE(status);
EXPECT_EQ(kValue, status.get().value());
EXPECT_EQ(kTestCid, channel_id);
}
TEST_F(ServiceMessageTest, PushChannelFailure) {
ExpectDefaultHandleMessage();
int kFlags = 123;
EXPECT_CALL(*endpoint(), PushChannel(message_.get(), kFlags, nullptr, _))
.WillOnce(Return(ByMove(ErrorStatus{EIO})));
int channel_id = -1;
auto status = message_->PushChannel(kFlags, nullptr, &channel_id);
ASSERT_FALSE(status);
EXPECT_EQ(EIO, status.error());
}
TEST_F(ServiceMessageTest, PushChannelDifferentService) {
ExpectDefaultHandleMessage();
auto endpoint2 = std::make_unique<testing::StrictMock<MockEndpoint>>();
EXPECT_CALL(*endpoint2, SetService(_))
.Times(2)
.WillRepeatedly(Return(Status<void>{}));
auto service2 =
std::make_shared<MockService>("MockSvc2", std::move(endpoint2));
int kFlags = 123;
int32_t kValue = 12;
EXPECT_CALL(*static_cast<MockEndpoint*>(service2->endpoint()),
PushChannel(message_.get(), kFlags, nullptr, _))
.WillOnce(DoAll(SetArgPointee<3>(kTestCid),
Return(ByMove(RemoteChannelHandle{kValue}))));
int channel_id = -1;
auto status =
message_->PushChannel(service2.get(), kFlags, nullptr, &channel_id);
ASSERT_TRUE(status);
EXPECT_EQ(kValue, status.get().value());
EXPECT_EQ(kTestCid, channel_id);
}
TEST_F(ServiceMessageTest, CheckChannelSameService) {
ExpectDefaultHandleMessage();
auto test_channel = std::make_shared<Channel>();
ChannelReference kRef = 123;
EXPECT_CALL(*endpoint(), CheckChannel(message_.get(), kRef, _))
.WillOnce(DoAll(SetArgPointee<2>(test_channel.get()), Return(kTestCid)));
std::shared_ptr<Channel> channel;
auto status = message_->CheckChannel(kRef, &channel);
ASSERT_TRUE(status);
EXPECT_EQ(kTestCid, status.get());
EXPECT_EQ(test_channel, channel);
}
TEST_F(ServiceMessageTest, CheckChannelFailure) {
ExpectDefaultHandleMessage();
ChannelReference kRef = 123;
EXPECT_CALL(*endpoint(), CheckChannel(message_.get(), kRef, _))
.WillOnce(Return(ByMove(ErrorStatus{EOPNOTSUPP})));
std::shared_ptr<Channel> channel;
auto status = message_->CheckChannel(kRef, &channel);
ASSERT_FALSE(status);
EXPECT_EQ(EOPNOTSUPP, status.error());
}
TEST_F(ServiceMessageTest, CheckChannelDifferentService) {
ExpectDefaultHandleMessage();
auto endpoint2 = std::make_unique<testing::StrictMock<MockEndpoint>>();
EXPECT_CALL(*endpoint2, SetService(_))
.Times(2)
.WillRepeatedly(Return(Status<void>{}));
auto service2 =
std::make_shared<MockService>("MockSvc2", std::move(endpoint2));
auto test_channel = std::make_shared<Channel>();
ChannelReference kRef = 123;
EXPECT_CALL(*static_cast<MockEndpoint*>(service2->endpoint()),
CheckChannel(message_.get(), kRef, _))
.WillOnce(DoAll(SetArgPointee<2>(test_channel.get()), Return(kTestCid)));
std::shared_ptr<Channel> channel;
auto status = message_->CheckChannel(service2.get(), kRef, &channel);
ASSERT_TRUE(status);
EXPECT_EQ(kTestCid, status.get());
EXPECT_EQ(test_channel, channel);
}