avs-device-sdk/CertifiedSender/test/CertifiedSenderTest.cpp

414 lines
17 KiB
C++

/*
* Copyright Amazon.com, Inc. or its affiliates. All Rights Reserved.
*
* Licensed under the Apache License, Version 2.0 (the "License").
* You may not use this file except in compliance with the License.
* A copy of the License is located at
*
* http://aws.amazon.com/apache2.0/
*
* or in the "license" file accompanying this file. This file is distributed
* on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either
* express or implied. See the License for the specific language governing
* permissions and limitations under the License.
*/
#include <chrono>
#include <memory>
#include <queue>
#include <gtest/gtest.h>
#include <AVSCommon/AVS/AbstractAVSConnectionManager.h>
#include <AVSCommon/AVS/Initialization/AlexaClientSDKInit.h>
#include <AVSCommon/SDKInterfaces/ConnectionStatusObserverInterface.h>
#include <AVSCommon/SDKInterfaces/MessageRequestObserverInterface.h>
#include <AVSCommon/SDKInterfaces/MockMessageSender.h>
#include <AVSCommon/Utils/PromiseFuturePair.h>
#include <RegistrationManager/MockCustomerDataManager.h>
#include "CertifiedSender/CertifiedSender.h"
using namespace ::testing;
namespace alexaClientSDK {
namespace certifiedSender {
namespace test {
using namespace avsCommon::sdkInterfaces;
/// A sample message
static const std::string TEST_MESSAGE = "TEST_MESSAGE";
/// A sample message URI.
static const std::string TEST_URI = "TEST_URI";
/// Timeout used in test
static const auto TEST_TIMEOUT = std::chrono::seconds(5);
/// Very long timeout used in test
static const auto LONG_TEST_TIMEOUT = std::chrono::seconds(20);
class MockConnection : public avsCommon::avs::AbstractAVSConnectionManager {
MOCK_METHOD0(enable, void());
MOCK_METHOD0(disable, void());
MOCK_METHOD0(isEnabled, bool());
MOCK_METHOD0(reconnect, void());
MOCK_CONST_METHOD0(isConnected, bool());
MOCK_METHOD0(onWakeConnectionRetry, void());
MOCK_METHOD1(
addMessageObserver,
void(std::shared_ptr<avsCommon::sdkInterfaces::MessageObserverInterface> observer));
MOCK_METHOD1(
removeMessageObserver,
void(std::shared_ptr<avsCommon::sdkInterfaces::MessageObserverInterface> observer));
MOCK_METHOD1(sendMessage, void(std::shared_ptr<avsCommon::avs::MessageRequest> request));
MOCK_METHOD1(setAVSGateway, void(const std::string& avsGateway));
MOCK_CONST_METHOD0(getAVSGateway, std::string());
};
class MockMessageStorage : public MessageStorageInterface {
public:
MOCK_METHOD0(createDatabase, bool());
MOCK_METHOD0(open, bool());
MOCK_METHOD0(close, void());
MOCK_METHOD2(store, bool(const std::string& message, int* id));
MOCK_METHOD3(store, bool(const std::string& message, const std::string& uriPathExtension, int* id));
MOCK_METHOD1(load, bool(std::queue<StoredMessage>* messageContainer));
MOCK_METHOD1(erase, bool(int messageId));
MOCK_METHOD0(clearDatabase, bool());
virtual ~MockMessageStorage() = default;
};
class CertifiedSenderTest : public ::testing::Test {
public:
protected:
void SetUp() override {
static const std::string CONFIGURATION = R"({
"certifiedSender" : {
"databaseFilePath":"database.db"
}
})";
auto configuration = std::shared_ptr<std::stringstream>(new std::stringstream());
(*configuration) << CONFIGURATION;
ASSERT_TRUE(avsCommon::avs::initialization::AlexaClientSDKInit::initialize({configuration}));
m_customerDataManager = std::make_shared<NiceMock<registrationManager::MockCustomerDataManager>>();
m_mockMessageSender = std::make_shared<avsCommon::sdkInterfaces::test::MockMessageSender>();
m_connection = std::make_shared<MockConnection>();
m_storage = std::make_shared<MockMessageStorage>();
EXPECT_CALL(*m_storage, open()).Times(1).WillOnce(Return(true));
EXPECT_CALL(*m_storage, load(_)).Times(1).WillOnce(Return(true));
m_certifiedSender =
CertifiedSender::create(m_mockMessageSender, m_connection, m_storage, m_customerDataManager);
}
void TearDown() override {
if (avsCommon::avs::initialization::AlexaClientSDKInit::isInitialized()) {
avsCommon::avs::initialization::AlexaClientSDKInit::uninitialize();
}
m_certifiedSender->shutdown();
}
/**
* Utility function to test that messages that receive a non-retryable status response are not retried.
* @param status The (non-retryable) status.
* @return Whether the test succeeded.
*/
bool testNotRetryable(MessageRequestObserverInterface::Status status);
/**
* Utility function to test that messages that receive a retryable status response are retried.
*
* @note A long timeout must be used because the first retry will happen only 10s after the first attempt.
* @param status The (retryable) status.
* @return Whether the test succeeded.
*/
bool testRetryable(MessageRequestObserverInterface::Status status);
/// Class under test.
std::shared_ptr<CertifiedSender> m_certifiedSender;
/// Mock message storage layer.
std::shared_ptr<MockMessageStorage> m_storage;
/// Pointer to connection. We need to remove certifiedSender as a connection observer or both objects will never
/// be deleted.
std::shared_ptr<MockConnection> m_connection;
/// The pointer to the customer data manager
std::shared_ptr<registrationManager::MockCustomerDataManager> m_customerDataManager;
/// The mock message sender instance.
std::shared_ptr<avsCommon::sdkInterfaces::test::MockMessageSender> m_mockMessageSender;
};
bool CertifiedSenderTest::testNotRetryable(MessageRequestObserverInterface::Status status) {
avsCommon::utils::PromiseFuturePair<bool> requestSent;
std::stringstream messageStream;
messageStream << "TestNotRetryableMessage-" << status;
std::string message = messageStream.str();
{
InSequence s;
EXPECT_CALL(*m_storage, store(_, _, _)).WillOnce(Return(true));
EXPECT_CALL(*m_mockMessageSender, sendMessage(_))
.Times(1)
.WillOnce(Invoke([status, &requestSent, message](std::shared_ptr<avsCommon::avs::MessageRequest> request) {
EXPECT_EQ(request->getJsonContent(), message);
request->sendCompleted(status);
requestSent.setValue(true);
}));
EXPECT_CALL(*m_storage, erase(_)).WillOnce(Return(true));
}
m_certifiedSender->sendJSONMessage(message, TEST_URI);
/// wait for requests to get sent out.
return requestSent.waitFor(TEST_TIMEOUT);
}
bool CertifiedSenderTest::testRetryable(MessageRequestObserverInterface::Status status) {
avsCommon::utils::PromiseFuturePair<bool> requestSent;
std::stringstream messageStream;
messageStream << "TestRetryableMessage-" << status;
std::string message = messageStream.str();
{
InSequence s;
EXPECT_CALL(*m_storage, store(_, _, _)).WillOnce(Return(true));
EXPECT_CALL(*m_mockMessageSender, sendMessage(_))
.Times(1)
.WillOnce(Invoke([status, message](std::shared_ptr<avsCommon::avs::MessageRequest> request) {
EXPECT_EQ(request->getJsonContent(), message);
request->sendCompleted(status);
}));
EXPECT_CALL(*m_mockMessageSender, sendMessage(_))
.Times(1)
.WillOnce(Invoke([&requestSent, message](std::shared_ptr<avsCommon::avs::MessageRequest> request) {
EXPECT_EQ(request->getJsonContent(), message);
request->sendCompleted(avsCommon::sdkInterfaces::MessageRequestObserverInterface::Status::SUCCESS);
requestSent.setValue(true);
}));
EXPECT_CALL(*m_storage, erase(_)).WillOnce(Return(true));
}
m_certifiedSender->sendJSONMessage(message, TEST_URI);
/// wait for requests to get sent out.
return requestSent.waitFor(LONG_TEST_TIMEOUT);
}
/**
* Check that @c clearData() method clears the persistent message storage and the current msg queue
*/
TEST_F(CertifiedSenderTest, test_clearData) {
EXPECT_CALL(*m_storage, clearDatabase()).Times(1);
m_certifiedSender->clearData();
}
/**
* Tests various failure scenarios for the init method.
*/
TEST_F(CertifiedSenderTest, test_initFailsWhenStorageMethodsFail) {
/// Test if the init method fails when createDatabase on storage fails.
{
EXPECT_CALL(*m_storage, open()).Times(1).WillOnce(Return(false));
EXPECT_CALL(*m_storage, createDatabase()).Times(1).WillOnce(Return(false));
EXPECT_CALL(*m_storage, load(_)).Times(0);
auto certifiedSender =
CertifiedSender::create(m_mockMessageSender, m_connection, m_storage, m_customerDataManager);
ASSERT_EQ(certifiedSender, nullptr);
}
/// Test if the init method fails when load from storage fails.
{
EXPECT_CALL(*m_storage, open()).Times(1).WillOnce(Return(true));
EXPECT_CALL(*m_storage, load(_)).Times(1).WillOnce(Return(false));
auto certifiedSender =
CertifiedSender::create(m_mockMessageSender, m_connection, m_storage, m_customerDataManager);
ASSERT_EQ(certifiedSender, nullptr);
}
}
/**
* Tests if the stored messages get sent when a connection is established.
*/
TEST_F(CertifiedSenderTest, testTimer_storedMessagesGetSent) {
EXPECT_CALL(*m_storage, open()).Times(1).WillOnce(Return(true));
/// Return messages from storage.
EXPECT_CALL(*m_storage, load(_))
.Times(1)
.WillOnce(Invoke([](std::queue<MessageStorageInterface::StoredMessage>* storedMessages) {
storedMessages->push(MessageStorageInterface::StoredMessage(1, "testMessage_1"));
storedMessages->push(MessageStorageInterface::StoredMessage(2, "testMessage_2"));
return true;
}));
avsCommon::utils::PromiseFuturePair<bool> allRequestsSent;
{
InSequence s;
EXPECT_CALL(*m_mockMessageSender, sendMessage(_))
.Times(1)
.WillOnce(Invoke([](std::shared_ptr<avsCommon::avs::MessageRequest> request) {
ASSERT_EQ(request->getJsonContent(), "testMessage_1");
request->sendCompleted(avsCommon::sdkInterfaces::MessageRequestObserverInterface::Status::SUCCESS);
}));
EXPECT_CALL(*m_storage, erase(1)).WillOnce(Return(true));
EXPECT_CALL(*m_mockMessageSender, sendMessage(_))
.Times(1)
.WillOnce(Invoke([&allRequestsSent](std::shared_ptr<avsCommon::avs::MessageRequest> request) {
ASSERT_EQ(request->getJsonContent(), "testMessage_2");
request->sendCompleted(avsCommon::sdkInterfaces::MessageRequestObserverInterface::Status::SUCCESS);
allRequestsSent.setValue(true);
}));
EXPECT_CALL(*m_storage, erase(2)).WillOnce(Return(true));
}
auto certifiedSender = CertifiedSender::create(m_mockMessageSender, m_connection, m_storage, m_customerDataManager);
std::static_pointer_cast<avsCommon::sdkInterfaces::ConnectionStatusObserverInterface>(certifiedSender)
->onConnectionStatusChanged(
avsCommon::sdkInterfaces::ConnectionStatusObserverInterface::Status::CONNECTED,
avsCommon::sdkInterfaces::ConnectionStatusObserverInterface::ChangedReason::SUCCESS);
/// wait for requests to get sent out.
EXPECT_TRUE(allRequestsSent.waitFor(TEST_TIMEOUT));
/// Cleanup
certifiedSender->shutdown();
}
/**
* Verify that a message with a URI specified will be sent out by the sender with the URI.
*/
TEST_F(CertifiedSenderTest, testTimer_SendMessageWithURI) {
avsCommon::utils::PromiseFuturePair<std::shared_ptr<avsCommon::avs::MessageRequest>> requestSent;
EXPECT_CALL(*m_storage, store(_, TEST_URI, _)).WillOnce(Return(true));
{
InSequence s;
EXPECT_CALL(*m_mockMessageSender, sendMessage(_))
.WillOnce(Invoke([&requestSent](std::shared_ptr<avsCommon::avs::MessageRequest> request) {
requestSent.setValue(request);
request->sendCompleted(avsCommon::sdkInterfaces::MessageRequestObserverInterface::Status::SUCCESS);
}));
EXPECT_CALL(*m_storage, erase(_)).WillOnce(Return(true));
}
std::static_pointer_cast<avsCommon::sdkInterfaces::ConnectionStatusObserverInterface>(m_certifiedSender)
->onConnectionStatusChanged(
avsCommon::sdkInterfaces::ConnectionStatusObserverInterface::Status::CONNECTED,
avsCommon::sdkInterfaces::ConnectionStatusObserverInterface::ChangedReason::SUCCESS);
m_certifiedSender->sendJSONMessage(TEST_MESSAGE, TEST_URI);
EXPECT_TRUE(requestSent.waitFor(TEST_TIMEOUT));
EXPECT_EQ(requestSent.getValue()->getJsonContent(), TEST_MESSAGE);
EXPECT_EQ(requestSent.getValue()->getUriPathExtension(), TEST_URI);
}
/**
* Tests if messages are re-submitted when the response is a re-tryable response.
*/
TEST_F(CertifiedSenderTest, testSlow_retryableResponsesAreRetried) {
std::static_pointer_cast<avsCommon::sdkInterfaces::ConnectionStatusObserverInterface>(m_certifiedSender)
->onConnectionStatusChanged(
avsCommon::sdkInterfaces::ConnectionStatusObserverInterface::Status::CONNECTED,
avsCommon::sdkInterfaces::ConnectionStatusObserverInterface::ChangedReason::SUCCESS);
{
SCOPED_TRACE(MessageRequestObserverInterface::Status::SERVER_INTERNAL_ERROR_V2);
EXPECT_TRUE(testRetryable(MessageRequestObserverInterface::Status::SERVER_INTERNAL_ERROR_V2));
}
{
SCOPED_TRACE(MessageRequestObserverInterface::Status::THROTTLED);
EXPECT_TRUE(testRetryable(MessageRequestObserverInterface::Status::THROTTLED));
}
{
SCOPED_TRACE(MessageRequestObserverInterface::Status::PENDING);
EXPECT_TRUE(testRetryable(MessageRequestObserverInterface::Status::PENDING));
}
{
SCOPED_TRACE(MessageRequestObserverInterface::Status::NOT_CONNECTED);
EXPECT_TRUE(testRetryable(MessageRequestObserverInterface::Status::NOT_CONNECTED));
}
{
SCOPED_TRACE(MessageRequestObserverInterface::Status::NOT_SYNCHRONIZED);
EXPECT_TRUE(testRetryable(MessageRequestObserverInterface::Status::NOT_SYNCHRONIZED));
}
{
SCOPED_TRACE(MessageRequestObserverInterface::Status::TIMEDOUT);
EXPECT_TRUE(testRetryable(MessageRequestObserverInterface::Status::TIMEDOUT));
}
{
SCOPED_TRACE(MessageRequestObserverInterface::Status::PROTOCOL_ERROR);
EXPECT_TRUE(testRetryable(MessageRequestObserverInterface::Status::PROTOCOL_ERROR));
}
{
SCOPED_TRACE(MessageRequestObserverInterface::Status::INTERNAL_ERROR);
EXPECT_TRUE(testRetryable(MessageRequestObserverInterface::Status::INTERNAL_ERROR));
}
{
SCOPED_TRACE(MessageRequestObserverInterface::Status::REFUSED);
EXPECT_TRUE(testRetryable(MessageRequestObserverInterface::Status::REFUSED));
}
{
SCOPED_TRACE(MessageRequestObserverInterface::Status::INVALID_AUTH);
EXPECT_TRUE(testRetryable(MessageRequestObserverInterface::Status::INVALID_AUTH));
}
}
/**
* Tests if messages are discarded when the response is a non-retryable response.
*/
TEST_F(CertifiedSenderTest, testSlow_nonRetryableResponsesAreNotRetried) {
std::static_pointer_cast<avsCommon::sdkInterfaces::ConnectionStatusObserverInterface>(m_certifiedSender)
->onConnectionStatusChanged(
avsCommon::sdkInterfaces::ConnectionStatusObserverInterface::Status::CONNECTED,
avsCommon::sdkInterfaces::ConnectionStatusObserverInterface::ChangedReason::SUCCESS);
{
SCOPED_TRACE(MessageRequestObserverInterface::Status::SUCCESS);
EXPECT_TRUE(testNotRetryable(MessageRequestObserverInterface::Status::SUCCESS));
}
{
SCOPED_TRACE(MessageRequestObserverInterface::Status::SUCCESS_ACCEPTED);
EXPECT_TRUE(testNotRetryable(MessageRequestObserverInterface::Status::SUCCESS_ACCEPTED));
}
{
SCOPED_TRACE(MessageRequestObserverInterface::Status::SUCCESS_NO_CONTENT);
EXPECT_TRUE(testNotRetryable(MessageRequestObserverInterface::Status::SUCCESS_NO_CONTENT));
}
{
SCOPED_TRACE(MessageRequestObserverInterface::Status::CANCELED);
EXPECT_TRUE(testNotRetryable(MessageRequestObserverInterface::Status::CANCELED));
}
{
SCOPED_TRACE(MessageRequestObserverInterface::Status::SERVER_OTHER_ERROR);
EXPECT_TRUE(testNotRetryable(MessageRequestObserverInterface::Status::SERVER_OTHER_ERROR));
}
{
SCOPED_TRACE(MessageRequestObserverInterface::Status::BAD_REQUEST);
EXPECT_TRUE(testNotRetryable(MessageRequestObserverInterface::Status::BAD_REQUEST));
}
}
} // namespace test
} // namespace certifiedSender
} // namespace alexaClientSDK