SmartAudio/package/avs/avs-sdk/files/avs-device-sdk/CapabilityAgents/TemplateRuntime/test/TemplateRuntimeTest.cpp

748 lines
34 KiB
C++

/*
* Copyright 2017-2018 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.
*/
/// @file TemplateRuntimeTest
#include <future>
#include <memory>
#include <gmock/gmock.h>
#include <gtest/gtest.h>
#include <rapidjson/document.h>
#include <AVSCommon/AVS/Attachment/MockAttachmentManager.h>
#include <AVSCommon/SDKInterfaces/AudioPlayerInterface.h>
#include <AVSCommon/SDKInterfaces/AudioPlayerObserverInterface.h>
#include <AVSCommon/SDKInterfaces/MockDirectiveHandlerResult.h>
#include <AVSCommon/SDKInterfaces/MockExceptionEncounteredSender.h>
#include <AVSCommon/SDKInterfaces/MockFocusManager.h>
#include <AVSCommon/SDKInterfaces/TemplateRuntimeObserverInterface.h>
#include <AVSCommon/Utils/JSON/JSONUtils.h>
#include <AVSCommon/Utils/Memory/Memory.h>
#include "TemplateRuntime/TemplateRuntime.h"
namespace alexaClientSDK {
namespace capabilityAgents {
namespace templateRuntime {
namespace test {
using namespace avsCommon::avs;
using namespace avsCommon::avs::attachment::test;
using namespace avsCommon::sdkInterfaces;
using namespace avsCommon::sdkInterfaces::test;
using namespace avsCommon::utils::memory;
using namespace rapidjson;
using namespace ::testing;
/// Timeout when waiting for futures to be set.
static std::chrono::milliseconds TIMEOUT(1000);
/// Timeout when waiting for clearTemplateCard.
static std::chrono::milliseconds TEMPLATE_TIMEOUT(5000);
/// Timeout when waiting for clearTemplateCard.
static std::chrono::milliseconds PLAYER_FINISHED_TIMEOUT(5000);
/// The namespace for this capability agent.
static const std::string NAMESPACE{"TemplateRuntime"};
/// An unknown directive signature.
static const std::string UNKNOWN_DIRECTIVE{"Unknown"};
/// The RenderTemplate directive signature.
static const NamespaceAndName TEMPLATE{NAMESPACE, "RenderTemplate"};
/// The RenderPlayerInfo directive signature.
static const NamespaceAndName PLAYER_INFO{NAMESPACE, "RenderPlayerInfo"};
/// The @c MessageId identifer.
static const std::string MESSAGE_ID("messageId");
/// An audioItemId for the RenderPlayerInfo directive.
static const std::string AUDIO_ITEM_ID("AudioItemId abcdefgh");
/// An audioItemId without a corresponding RenderPlayerInfo directive.
static const std::string AUDIO_ITEM_ID_1("AudioItemId 12345678");
/// A RenderTemplate directive payload.
// clang-format off
static const std::string TEMPLATE_PAYLOAD = "{"
"\"token\":\"TOKEN1\","
"\"type\":\"BodyTemplate1\","
"\"title\":{"
"\"mainTitle\":\"MAIN_TITLE\","
"\"subTitle\":\"SUB_TITLE\""
"}"
"}";
// clang-format on
/// A RenderPlayerInfo directive payload.
// clang-format off
static const std::string PLAYERINFO_PAYLOAD = "{"
"\"audioItemId\":\"" + AUDIO_ITEM_ID + "\","
"\"content\":{"
"\"title\":\"TITLE\","
"\"header\":\"HEADER\""
"}"
"}";
// clang-format on
/// A malformed RenderPlayerInfo directive payload.
// clang-format off
static const std::string MALFORM_PLAYERINFO_PAYLOAD = "{"
"\"audioItemId\"::::\"" + AUDIO_ITEM_ID + "\","
"\"content\":{{{{"
"\"title\":\"TITLE\","
"\"header\":\"HEADER\""
"}"
"}";
// clang-format on
class MockAudioPlayer : public AudioPlayerInterface {
public:
MOCK_METHOD1(addObserver, void(std::shared_ptr<avsCommon::sdkInterfaces::AudioPlayerObserverInterface> observer));
MOCK_METHOD1(
removeObserver,
void(std::shared_ptr<avsCommon::sdkInterfaces::AudioPlayerObserverInterface> observer));
MOCK_METHOD0(getAudioItemOffset, std::chrono::milliseconds());
};
class MockGui : public TemplateRuntimeObserverInterface {
public:
MOCK_METHOD2(renderTemplateCard, void(const std::string& jsonPayload, avsCommon::avs::FocusState focusState));
MOCK_METHOD0(clearTemplateCard, void());
MOCK_METHOD3(
renderPlayerInfoCard,
void(
const std::string& jsonPayload,
TemplateRuntimeObserverInterface::AudioPlayerInfo audioPlayerInfo,
avsCommon::avs::FocusState focusState));
MOCK_METHOD0(clearPlayerInfoCard, void());
};
/// Test harness for @c TemplateRuntime class.
class TemplateRuntimeTest : public ::testing::Test {
public:
/// Set up the test harness for running a test.
void SetUp() override;
/// Clean up the test harness after running a test.
void TearDown() override;
/// Function to set the promise and wake @c m_wakeSetCompleteFuture.
void wakeOnSetCompleted();
/// Function to set the promise and wake @c m_wakeRenderTemplateCardFuture.
void wakeOnRenderTemplateCard();
/// Function to set the promise and wake @c m_wakeRenderPlayerInfoCardFuture.
void wakeOnRenderPlayerInfoCard();
/// Function to set the promise and wake @c m_wakeClearTemplateCardFuture.
void wakeOnClearTemplateCard();
/// Function to set the promise and wake @c m_wakeClearPlayerInfoCardFuture.
void wakeOnClearPlayerInfoCard();
/// Function to set the promise and wake @c m_wakeReleaseChannelFuture.
void wakeOnReleaseChannel();
/// A constructor which initializes the promises and futures needed for the test class.
TemplateRuntimeTest() :
m_wakeSetCompletedPromise{},
m_wakeSetCompletedFuture{m_wakeSetCompletedPromise.get_future()},
m_wakeRenderTemplateCardPromise{},
m_wakeRenderTemplateCardFuture{m_wakeRenderTemplateCardPromise.get_future()},
m_wakeRenderPlayerInfoCardPromise{},
m_wakeRenderPlayerInfoCardFuture{m_wakeRenderPlayerInfoCardPromise.get_future()},
m_wakeClearTemplateCardPromise{},
m_wakeClearTemplateCardFuture{m_wakeClearTemplateCardPromise.get_future()},
m_wakeClearPlayerInfoCardPromise{},
m_wakeClearPlayerInfoCardFuture{m_wakeClearPlayerInfoCardPromise.get_future()},
m_wakeReleaseChannelPromise{},
m_wakeReleaseChannelFuture{m_wakeReleaseChannelPromise.get_future()} {
}
protected:
/// Promise to synchronize directive handling through setCompleted.
std::promise<void> m_wakeSetCompletedPromise;
/// Future to synchronize directive handling through setCompleted.
std::future<void> m_wakeSetCompletedFuture;
/// Promise to synchronize directive handling with RenderTemplateCard callback.
std::promise<void> m_wakeRenderTemplateCardPromise;
/// Future to synchronize directive handling with RenderTemplateCard callback.
std::future<void> m_wakeRenderTemplateCardFuture;
/// Promise to synchronize directive handling with RenderPlayerInfoCard callback.
std::promise<void> m_wakeRenderPlayerInfoCardPromise;
/// Future to synchronize directive handling with RenderPlayerInfoCard callback.
std::future<void> m_wakeRenderPlayerInfoCardFuture;
/// Promise to synchronize ClearTemplateCard callback.
std::promise<void> m_wakeClearTemplateCardPromise;
/// Future to synchronize ClearTemplateCard callback.
std::future<void> m_wakeClearTemplateCardFuture;
/// Promise to synchronize ClearPlayerInfoCard callback.
std::promise<void> m_wakeClearPlayerInfoCardPromise;
/// Future to synchronize ClearPlayerInfoCard callback.
std::future<void> m_wakeClearPlayerInfoCardFuture;
/// Promise to synchronize releaseChannel calls.
std::promise<void> m_wakeReleaseChannelPromise;
/// Future to synchronize releaseChannel calls.
std::future<void> m_wakeReleaseChannelFuture;
/// A nice mock for the AudioPlayerInterface calls.
std::shared_ptr<NiceMock<MockAudioPlayer>> m_mockAudioPlayerInterface;
/// A strict mock that allows the test to strictly monitor the exceptions being sent.
std::shared_ptr<StrictMock<MockExceptionEncounteredSender>> m_mockExceptionSender;
/// A strict mock that allows the test to strictly monitor the handling of directives.
std::unique_ptr<StrictMock<MockDirectiveHandlerResult>> m_mockDirectiveHandlerResult;
/// @c FocusManager to request focus to the Visual channel.
std::shared_ptr<MockFocusManager> m_mockFocusManager;
/// A strict mock to allow testing of the observer callback.
std::shared_ptr<StrictMock<MockGui>> m_mockGui;
/// A pointer to an instance of the TemplateRuntime that will be instantiated per test.
std::shared_ptr<TemplateRuntime> m_templateRuntime;
};
void TemplateRuntimeTest::SetUp() {
m_mockExceptionSender = std::make_shared<StrictMock<MockExceptionEncounteredSender>>();
m_mockDirectiveHandlerResult = make_unique<StrictMock<MockDirectiveHandlerResult>>();
m_mockFocusManager = std::make_shared<NiceMock<MockFocusManager>>();
m_mockAudioPlayerInterface = std::make_shared<NiceMock<MockAudioPlayer>>();
m_mockGui = std::make_shared<StrictMock<MockGui>>();
m_templateRuntime = TemplateRuntime::create(m_mockAudioPlayerInterface, m_mockFocusManager, m_mockExceptionSender);
m_templateRuntime->addObserver(m_mockGui);
ON_CALL(*m_mockFocusManager, acquireChannel(_, _, _)).WillByDefault(InvokeWithoutArgs([this] {
m_templateRuntime->onFocusChanged(avsCommon::avs::FocusState::FOREGROUND);
return true;
}));
ON_CALL(*m_mockFocusManager, releaseChannel(_, _)).WillByDefault(InvokeWithoutArgs([this] {
auto releaseChannelSuccess = std::make_shared<std::promise<bool>>();
std::future<bool> returnValue = releaseChannelSuccess->get_future();
m_templateRuntime->onFocusChanged(avsCommon::avs::FocusState::NONE);
releaseChannelSuccess->set_value(true);
return returnValue;
}));
}
void TemplateRuntimeTest::TearDown() {
if (m_templateRuntime) {
m_templateRuntime->shutdown();
m_templateRuntime.reset();
}
}
void TemplateRuntimeTest::wakeOnSetCompleted() {
m_wakeSetCompletedPromise.set_value();
}
void TemplateRuntimeTest::wakeOnRenderTemplateCard() {
m_wakeRenderTemplateCardPromise.set_value();
}
void TemplateRuntimeTest::wakeOnRenderPlayerInfoCard() {
m_wakeRenderPlayerInfoCardPromise.set_value();
}
void TemplateRuntimeTest::wakeOnClearTemplateCard() {
m_wakeClearTemplateCardPromise.set_value();
}
void TemplateRuntimeTest::wakeOnClearPlayerInfoCard() {
m_wakeClearPlayerInfoCardPromise.set_value();
}
void TemplateRuntimeTest::wakeOnReleaseChannel() {
m_wakeReleaseChannelPromise.set_value();
}
/**
* Tests creating the TemplateRuntime with a null audioPlayerInterface.
*/
TEST_F(TemplateRuntimeTest, testNullAudioPlayerInterface) {
auto templateRuntime = TemplateRuntime::create(nullptr, m_mockFocusManager, m_mockExceptionSender);
ASSERT_EQ(templateRuntime, nullptr);
}
/**
* Tests creating the TemplateRuntime with a null focusManagerInterface.
*/
TEST_F(TemplateRuntimeTest, testNullFocusManagerInterface) {
auto templateRuntime = TemplateRuntime::create(m_mockAudioPlayerInterface, nullptr, m_mockExceptionSender);
ASSERT_EQ(templateRuntime, nullptr);
}
/**
* Tests creating the TemplateRuntime with a null exceptionSender.
*/
TEST_F(TemplateRuntimeTest, testNullExceptionSender) {
auto templateRuntime = TemplateRuntime::create(m_mockAudioPlayerInterface, m_mockFocusManager, nullptr);
ASSERT_EQ(templateRuntime, nullptr);
}
/**
* Tests that the TemplateRuntime successfully add itself with the AudioPlayer at constructor time, and
* successfully remove itself with the AudioPlayer during shutdown.
*/
TEST_F(TemplateRuntimeTest, testAudioPlayerAddRemoveObserver) {
auto mockAudioPlayerInterface = std::make_shared<NiceMock<MockAudioPlayer>>();
auto mockExceptionSender = std::make_shared<StrictMock<MockExceptionEncounteredSender>>();
auto mockFocusManager = std::make_shared<NiceMock<MockFocusManager>>();
EXPECT_CALL(*mockAudioPlayerInterface, addObserver(NotNull())).Times(Exactly(1));
EXPECT_CALL(*mockAudioPlayerInterface, removeObserver(NotNull())).Times(Exactly(1));
auto templateRuntime = TemplateRuntime::create(mockAudioPlayerInterface, mockFocusManager, mockExceptionSender);
templateRuntime->shutdown();
}
/**
* Tests unknown Directive. Expect that the sendExceptionEncountered and setFailed will be called.
*/
TEST_F(TemplateRuntimeTest, testUnknownDirective) {
// Create Directive.
auto attachmentManager = std::make_shared<StrictMock<MockAttachmentManager>>();
auto avsMessageHeader = std::make_shared<AVSMessageHeader>(NAMESPACE, UNKNOWN_DIRECTIVE, MESSAGE_ID);
std::shared_ptr<AVSDirective> directive = AVSDirective::create("", avsMessageHeader, "", attachmentManager, "");
EXPECT_CALL(*m_mockExceptionSender, sendExceptionEncountered(_, _, _)).Times(Exactly(1));
EXPECT_CALL(*m_mockDirectiveHandlerResult, setFailed(_))
.Times(Exactly(1))
.WillOnce(InvokeWithoutArgs(this, &TemplateRuntimeTest::wakeOnSetCompleted));
m_templateRuntime->CapabilityAgent::preHandleDirective(directive, std::move(m_mockDirectiveHandlerResult));
m_wakeSetCompletedFuture.wait_for(TIMEOUT);
}
/**
* Tests RenderTemplate Directive. Expect that the renderTemplateCard callback will be called and clearTemplateCard will
* be called after 2s after DialogUXState is changed to IDLE state.
*/
TEST_F(TemplateRuntimeTest, testRenderTemplateDirective) {
// Create Directive.
auto attachmentManager = std::make_shared<StrictMock<MockAttachmentManager>>();
auto avsMessageHeader = std::make_shared<AVSMessageHeader>(TEMPLATE.nameSpace, TEMPLATE.name, MESSAGE_ID);
std::shared_ptr<AVSDirective> directive =
AVSDirective::create("", avsMessageHeader, TEMPLATE_PAYLOAD, attachmentManager, "");
EXPECT_CALL(*m_mockGui, renderTemplateCard(TEMPLATE_PAYLOAD, _))
.Times(Exactly(1))
.WillOnce(InvokeWithoutArgs(this, &TemplateRuntimeTest::wakeOnRenderTemplateCard));
EXPECT_CALL(*m_mockDirectiveHandlerResult, setCompleted())
.Times(Exactly(1))
.WillOnce(InvokeWithoutArgs(this, &TemplateRuntimeTest::wakeOnSetCompleted));
EXPECT_CALL(*m_mockGui, clearTemplateCard())
.Times(Exactly(1))
.WillOnce(InvokeWithoutArgs(this, &TemplateRuntimeTest::wakeOnClearTemplateCard));
m_templateRuntime->CapabilityAgent::preHandleDirective(directive, std::move(m_mockDirectiveHandlerResult));
m_templateRuntime->CapabilityAgent::handleDirective(MESSAGE_ID);
m_wakeSetCompletedFuture.wait_for(TIMEOUT);
m_wakeRenderTemplateCardFuture.wait_for(TIMEOUT);
m_templateRuntime->onDialogUXStateChanged(
avsCommon::sdkInterfaces::DialogUXStateObserverInterface::DialogUXState::IDLE);
m_wakeClearTemplateCardFuture.wait_for(TEMPLATE_TIMEOUT);
}
/**
* Tests RenderTemplate Directive using the handleDirectiveImmediately. Expect that the renderTemplateCard
* callback will be called.
*/
TEST_F(TemplateRuntimeTest, testHandleDirectiveImmediately) {
// Create Directive.
auto attachmentManager = std::make_shared<StrictMock<MockAttachmentManager>>();
auto avsMessageHeader = std::make_shared<AVSMessageHeader>(TEMPLATE.nameSpace, TEMPLATE.name, MESSAGE_ID);
std::shared_ptr<AVSDirective> directive =
AVSDirective::create("", avsMessageHeader, TEMPLATE_PAYLOAD, attachmentManager, "");
EXPECT_CALL(*m_mockGui, renderTemplateCard(TEMPLATE_PAYLOAD, _))
.Times(Exactly(1))
.WillOnce(InvokeWithoutArgs(this, &TemplateRuntimeTest::wakeOnRenderTemplateCard));
m_templateRuntime->handleDirectiveImmediately(directive);
m_wakeRenderTemplateCardFuture.wait_for(TIMEOUT);
}
/**
* Tests RenderTemplate Directive received before the corresponding AudioPlayer call. Expect
* that the renderTemplateCard callback will be called and clearPlayerInfoCard will be called after 2s after Audio State
* is changed to FINISHED state.
*/
TEST_F(TemplateRuntimeTest, testRenderPlayerInfoDirectiveBefore) {
// Create Directive.
auto attachmentManager = std::make_shared<StrictMock<MockAttachmentManager>>();
auto avsMessageHeader = std::make_shared<AVSMessageHeader>(PLAYER_INFO.nameSpace, PLAYER_INFO.name, MESSAGE_ID);
std::shared_ptr<AVSDirective> directive =
AVSDirective::create("", avsMessageHeader, PLAYERINFO_PAYLOAD, attachmentManager, "");
::testing::InSequence s;
EXPECT_CALL(*m_mockDirectiveHandlerResult, setCompleted())
.Times(Exactly(1))
.WillOnce(InvokeWithoutArgs(this, &TemplateRuntimeTest::wakeOnSetCompleted));
EXPECT_CALL(*m_mockGui, renderTemplateCard(_, _)).Times(Exactly(0));
// do not expect renderPlayerInfo card call until AudioPlayer notify with the correct audioItemId
EXPECT_CALL(*m_mockGui, renderPlayerInfoCard(_, _, _)).Times(Exactly(0));
m_templateRuntime->CapabilityAgent::preHandleDirective(directive, std::move(m_mockDirectiveHandlerResult));
m_templateRuntime->CapabilityAgent::handleDirective(MESSAGE_ID);
m_wakeSetCompletedFuture.wait_for(TIMEOUT);
EXPECT_CALL(*m_mockGui, renderPlayerInfoCard(PLAYERINFO_PAYLOAD, _, _))
.Times(Exactly(2))
.WillOnce(InvokeWithoutArgs(this, &TemplateRuntimeTest::wakeOnRenderPlayerInfoCard))
.WillOnce(InvokeWithoutArgs([] {}));
AudioPlayerObserverInterface::Context context;
context.audioItemId = AUDIO_ITEM_ID;
context.offset = TIMEOUT;
m_templateRuntime->onPlayerActivityChanged(avsCommon::avs::PlayerActivity::PLAYING, context);
m_wakeRenderPlayerInfoCardFuture.wait_for(TIMEOUT);
EXPECT_CALL(*m_mockGui, clearPlayerInfoCard())
.Times(Exactly(1))
.WillOnce(InvokeWithoutArgs(this, &TemplateRuntimeTest::wakeOnClearPlayerInfoCard));
m_templateRuntime->onPlayerActivityChanged(avsCommon::avs::PlayerActivity::FINISHED, context);
m_wakeClearPlayerInfoCardFuture.wait_for(PLAYER_FINISHED_TIMEOUT);
}
/**
* Tests RenderTemplate Directive received after the corresponding AudioPlayer call. Expect
* that the renderTemplateCard callback will be called.
*/
TEST_F(TemplateRuntimeTest, testRenderPlayerInfoDirectiveAfter) {
// Create Directive.
auto attachmentManager = std::make_shared<StrictMock<MockAttachmentManager>>();
auto avsMessageHeader = std::make_shared<AVSMessageHeader>(PLAYER_INFO.nameSpace, PLAYER_INFO.name, MESSAGE_ID);
std::shared_ptr<AVSDirective> directive =
AVSDirective::create("", avsMessageHeader, PLAYERINFO_PAYLOAD, attachmentManager, "");
EXPECT_CALL(*m_mockGui, renderPlayerInfoCard(PLAYERINFO_PAYLOAD, _, _))
.Times(Exactly(1))
.WillOnce(InvokeWithoutArgs(this, &TemplateRuntimeTest::wakeOnRenderPlayerInfoCard));
EXPECT_CALL(*m_mockDirectiveHandlerResult, setCompleted())
.Times(Exactly(1))
.WillOnce(InvokeWithoutArgs(this, &TemplateRuntimeTest::wakeOnSetCompleted));
AudioPlayerObserverInterface::Context context;
context.audioItemId = AUDIO_ITEM_ID;
context.offset = TIMEOUT;
m_templateRuntime->onPlayerActivityChanged(avsCommon::avs::PlayerActivity::PLAYING, context);
m_templateRuntime->CapabilityAgent::preHandleDirective(directive, std::move(m_mockDirectiveHandlerResult));
m_templateRuntime->CapabilityAgent::handleDirective(MESSAGE_ID);
m_wakeRenderPlayerInfoCardFuture.wait_for(TIMEOUT);
m_wakeSetCompletedFuture.wait_for(TIMEOUT);
}
/**
* Tests RenderTemplate Directive received without an audioItemId. Expect that the
* sendExceptionEncountered and setFailed will be called.
*/
TEST_F(TemplateRuntimeTest, testRenderPlayerInfoDirectiveWithoutAudioItemId) {
// Create Directive.
auto attachmentManager = std::make_shared<StrictMock<MockAttachmentManager>>();
auto avsMessageHeader = std::make_shared<AVSMessageHeader>(PLAYER_INFO.nameSpace, PLAYER_INFO.name, MESSAGE_ID);
std::shared_ptr<AVSDirective> directive =
AVSDirective::create("", avsMessageHeader, TEMPLATE_PAYLOAD, attachmentManager, "");
EXPECT_CALL(*m_mockExceptionSender, sendExceptionEncountered(_, _, _)).Times(Exactly(1));
EXPECT_CALL(*m_mockDirectiveHandlerResult, setFailed(_))
.Times(Exactly(1))
.WillOnce(InvokeWithoutArgs(this, &TemplateRuntimeTest::wakeOnSetCompleted));
m_templateRuntime->CapabilityAgent::preHandleDirective(directive, std::move(m_mockDirectiveHandlerResult));
m_wakeSetCompletedFuture.wait_for(TIMEOUT);
}
/**
* Tests when a malformed RenderTemplate Directive is received. Expect that the
* sendExceptionEncountered and setFailed will be called.
*/
TEST_F(TemplateRuntimeTest, testMalformedRenderPlayerInfoDirective) {
// Create Directive.
auto attachmentManager = std::make_shared<StrictMock<MockAttachmentManager>>();
auto avsMessageHeader = std::make_shared<AVSMessageHeader>(PLAYER_INFO.nameSpace, PLAYER_INFO.name, MESSAGE_ID);
std::shared_ptr<AVSDirective> directive =
AVSDirective::create("", avsMessageHeader, MALFORM_PLAYERINFO_PAYLOAD, attachmentManager, "");
EXPECT_CALL(*m_mockExceptionSender, sendExceptionEncountered(_, _, _)).Times(Exactly(1));
EXPECT_CALL(*m_mockDirectiveHandlerResult, setFailed(_))
.Times(Exactly(1))
.WillOnce(InvokeWithoutArgs(this, &TemplateRuntimeTest::wakeOnSetCompleted));
m_templateRuntime->CapabilityAgent::preHandleDirective(directive, std::move(m_mockDirectiveHandlerResult));
m_wakeSetCompletedFuture.wait_for(TIMEOUT);
}
/**
* Tests AudioPlayer notified the handling of AUDIO_ITEM_ID_1, and then RenderTemplate Directive with
* AUDIO_ITEM_ID is received. Expect that the renderTemplateCard callback will not be called until
* the AudioPlayer notified the handling of AUDIO_ITEM_ID later.
*/
TEST_F(TemplateRuntimeTest, testRenderPlayerInfoDirectiveDifferentAudioItemId) {
// Create Directive.
auto attachmentManager = std::make_shared<StrictMock<MockAttachmentManager>>();
auto avsMessageHeader = std::make_shared<AVSMessageHeader>(PLAYER_INFO.nameSpace, PLAYER_INFO.name, MESSAGE_ID);
std::shared_ptr<AVSDirective> directive =
AVSDirective::create("", avsMessageHeader, PLAYERINFO_PAYLOAD, attachmentManager, "");
EXPECT_CALL(*m_mockGui, renderPlayerInfoCard(PLAYERINFO_PAYLOAD, _, _)).Times(Exactly(0));
EXPECT_CALL(*m_mockDirectiveHandlerResult, setCompleted())
.Times(Exactly(1))
.WillOnce(InvokeWithoutArgs(this, &TemplateRuntimeTest::wakeOnSetCompleted));
AudioPlayerObserverInterface::Context context;
context.audioItemId = AUDIO_ITEM_ID_1;
context.offset = TIMEOUT;
m_templateRuntime->onPlayerActivityChanged(avsCommon::avs::PlayerActivity::PLAYING, context);
m_templateRuntime->CapabilityAgent::preHandleDirective(directive, std::move(m_mockDirectiveHandlerResult));
m_templateRuntime->CapabilityAgent::handleDirective(MESSAGE_ID);
m_wakeSetCompletedFuture.wait_for(TIMEOUT);
EXPECT_CALL(*m_mockGui, renderPlayerInfoCard(PLAYERINFO_PAYLOAD, _, _))
.Times(Exactly(1))
.WillOnce(InvokeWithoutArgs(this, &TemplateRuntimeTest::wakeOnRenderPlayerInfoCard));
context.audioItemId = AUDIO_ITEM_ID;
m_templateRuntime->onPlayerActivityChanged(avsCommon::avs::PlayerActivity::PLAYING, context);
m_wakeRenderPlayerInfoCardFuture.wait_for(TIMEOUT);
}
/**
* Tests AudioPlayer callbacks will trigger the correct renderPlayerInfoCard callbacks. Expect
* the payload, audioPlayerState and offset to match to the ones passed in by the
* AudioPlayerObserverInterface.
*/
TEST_F(TemplateRuntimeTest, testRenderPlayerInfoDirectiveAudioStateUpdate) {
// Create Directive.
auto attachmentManager = std::make_shared<StrictMock<MockAttachmentManager>>();
auto avsMessageHeader = std::make_shared<AVSMessageHeader>(PLAYER_INFO.nameSpace, PLAYER_INFO.name, MESSAGE_ID);
std::shared_ptr<AVSDirective> directive =
AVSDirective::create("", avsMessageHeader, PLAYERINFO_PAYLOAD, attachmentManager, "");
::testing::InSequence s;
// Send a directive first to TemplateRuntime
EXPECT_CALL(*m_mockDirectiveHandlerResult, setCompleted())
.Times(Exactly(1))
.WillOnce(InvokeWithoutArgs(this, &TemplateRuntimeTest::wakeOnSetCompleted));
m_templateRuntime->CapabilityAgent::preHandleDirective(directive, std::move(m_mockDirectiveHandlerResult));
m_templateRuntime->CapabilityAgent::handleDirective(MESSAGE_ID);
m_wakeSetCompletedFuture.wait_for(TIMEOUT);
AudioPlayerObserverInterface::Context context;
context.audioItemId = AUDIO_ITEM_ID;
// Test onAudioPlayed() callback with 100ms offset
std::promise<void> wakePlayPromise;
std::future<void> wakePlayFuture = wakePlayPromise.get_future();
context.offset = std::chrono::milliseconds(100);
EXPECT_CALL(*m_mockGui, renderPlayerInfoCard(PLAYERINFO_PAYLOAD, _, _))
.Times(Exactly(1))
.WillOnce(Invoke([&wakePlayPromise, context](
const std::string& jsonPayload,
TemplateRuntimeObserverInterface::AudioPlayerInfo audioPlayerInfo,
avsCommon::avs::FocusState focusState) {
EXPECT_EQ(audioPlayerInfo.audioPlayerState, avsCommon::avs::PlayerActivity::PLAYING);
EXPECT_EQ(audioPlayerInfo.offset, context.offset);
wakePlayPromise.set_value();
}));
m_templateRuntime->onPlayerActivityChanged(avsCommon::avs::PlayerActivity::PLAYING, context);
wakePlayFuture.wait_for(TIMEOUT);
// Test onAudioPaused() callback with 200ms offset
std::promise<void> wakePausePromise;
std::future<void> wakePauseFuture = wakePausePromise.get_future();
context.offset = std::chrono::milliseconds(200);
EXPECT_CALL(*m_mockGui, renderPlayerInfoCard(PLAYERINFO_PAYLOAD, _, _))
.Times(Exactly(1))
.WillOnce(Invoke([&wakePausePromise, context](
const std::string& jsonPayload,
TemplateRuntimeObserverInterface::AudioPlayerInfo audioPlayerInfo,
avsCommon::avs::FocusState focusState) {
EXPECT_EQ(audioPlayerInfo.audioPlayerState, avsCommon::avs::PlayerActivity::PAUSED);
EXPECT_EQ(audioPlayerInfo.offset, context.offset);
wakePausePromise.set_value();
}));
m_templateRuntime->onPlayerActivityChanged(avsCommon::avs::PlayerActivity::PAUSED, context);
wakePauseFuture.wait_for(TIMEOUT);
// Test onAudioStopped() callback with 300ms offset
std::promise<void> wakeStopPromise;
std::future<void> wakeStopFuture = wakeStopPromise.get_future();
context.offset = std::chrono::milliseconds(300);
EXPECT_CALL(*m_mockGui, renderPlayerInfoCard(PLAYERINFO_PAYLOAD, _, _))
.Times(Exactly(1))
.WillOnce(Invoke([&wakeStopPromise, context](
const std::string& jsonPayload,
TemplateRuntimeObserverInterface::AudioPlayerInfo audioPlayerInfo,
avsCommon::avs::FocusState focusState) {
EXPECT_EQ(audioPlayerInfo.audioPlayerState, avsCommon::avs::PlayerActivity::STOPPED);
EXPECT_EQ(audioPlayerInfo.offset, context.offset);
wakeStopPromise.set_value();
}));
m_templateRuntime->onPlayerActivityChanged(avsCommon::avs::PlayerActivity::STOPPED, context);
wakeStopFuture.wait_for(TIMEOUT);
// Test onAudioFinished() callback with 400ms offset
std::promise<void> wakeFinishPromise;
std::future<void> wakeFinishFuture = wakeFinishPromise.get_future();
context.offset = std::chrono::milliseconds(400);
EXPECT_CALL(*m_mockGui, renderPlayerInfoCard(PLAYERINFO_PAYLOAD, _, _))
.Times(Exactly(1))
.WillOnce(Invoke([&wakeFinishPromise, context](
const std::string& jsonPayload,
TemplateRuntimeObserverInterface::AudioPlayerInfo audioPlayerInfo,
avsCommon::avs::FocusState focusState) {
EXPECT_EQ(audioPlayerInfo.audioPlayerState, avsCommon::avs::PlayerActivity::FINISHED);
EXPECT_EQ(audioPlayerInfo.offset, context.offset);
wakeFinishPromise.set_value();
}));
m_templateRuntime->onPlayerActivityChanged(avsCommon::avs::PlayerActivity::FINISHED, context);
wakeFinishFuture.wait_for(TIMEOUT);
}
/**
* Tests that if focus is changed to none, the clearTemplateCard() will be called.
*/
TEST_F(TemplateRuntimeTest, testFocusNone) {
// Create Directive.
auto attachmentManager = std::make_shared<StrictMock<MockAttachmentManager>>();
auto avsMessageHeader = std::make_shared<AVSMessageHeader>(TEMPLATE.nameSpace, TEMPLATE.name, MESSAGE_ID);
std::shared_ptr<AVSDirective> directive =
AVSDirective::create("", avsMessageHeader, TEMPLATE_PAYLOAD, attachmentManager, "");
EXPECT_CALL(*m_mockGui, renderTemplateCard(TEMPLATE_PAYLOAD, _))
.Times(Exactly(1))
.WillOnce(InvokeWithoutArgs(this, &TemplateRuntimeTest::wakeOnRenderTemplateCard));
EXPECT_CALL(*m_mockDirectiveHandlerResult, setCompleted())
.Times(Exactly(1))
.WillOnce(InvokeWithoutArgs(this, &TemplateRuntimeTest::wakeOnSetCompleted));
EXPECT_CALL(*m_mockGui, clearTemplateCard())
.Times(Exactly(1))
.WillOnce(InvokeWithoutArgs(this, &TemplateRuntimeTest::wakeOnClearTemplateCard));
m_templateRuntime->CapabilityAgent::preHandleDirective(directive, std::move(m_mockDirectiveHandlerResult));
m_templateRuntime->CapabilityAgent::handleDirective(MESSAGE_ID);
m_wakeSetCompletedFuture.wait_for(TIMEOUT);
m_wakeRenderTemplateCardFuture.wait_for(TIMEOUT);
m_templateRuntime->onFocusChanged(FocusState::NONE);
m_wakeClearTemplateCardFuture.wait_for(TIMEOUT);
}
/**
* Tests that if displayCardCleared() is called, the clearTemplateCard() will not be called.
*/
TEST_F(TemplateRuntimeTest, testDisplayCardCleared) {
// Create Directive.
auto attachmentManager = std::make_shared<StrictMock<MockAttachmentManager>>();
auto avsMessageHeader = std::make_shared<AVSMessageHeader>(TEMPLATE.nameSpace, TEMPLATE.name, MESSAGE_ID);
std::shared_ptr<AVSDirective> directive =
AVSDirective::create("", avsMessageHeader, TEMPLATE_PAYLOAD, attachmentManager, "");
EXPECT_CALL(*m_mockGui, renderTemplateCard(TEMPLATE_PAYLOAD, _))
.Times(Exactly(1))
.WillOnce(InvokeWithoutArgs(this, &TemplateRuntimeTest::wakeOnRenderTemplateCard));
EXPECT_CALL(*m_mockDirectiveHandlerResult, setCompleted())
.Times(Exactly(1))
.WillOnce(InvokeWithoutArgs(this, &TemplateRuntimeTest::wakeOnSetCompleted));
EXPECT_CALL(*m_mockGui, clearTemplateCard()).Times(Exactly(0));
EXPECT_CALL(*m_mockFocusManager, releaseChannel(_, _)).Times(Exactly(1)).WillOnce(InvokeWithoutArgs([this] {
auto releaseChannelSuccess = std::make_shared<std::promise<bool>>();
std::future<bool> returnValue = releaseChannelSuccess->get_future();
m_templateRuntime->onFocusChanged(avsCommon::avs::FocusState::NONE);
releaseChannelSuccess->set_value(true);
wakeOnReleaseChannel();
return returnValue;
}));
m_templateRuntime->CapabilityAgent::preHandleDirective(directive, std::move(m_mockDirectiveHandlerResult));
m_templateRuntime->CapabilityAgent::handleDirective(MESSAGE_ID);
m_wakeSetCompletedFuture.wait_for(TIMEOUT);
m_wakeRenderTemplateCardFuture.wait_for(TIMEOUT);
m_templateRuntime->displayCardCleared();
m_wakeReleaseChannelFuture.wait_for(TIMEOUT);
}
/**
* Tests that if another displayCard event is sent before channel's focus is set to none, the state machine would
* transition to REACQUIRING state and acquireChannel again to display the card.
*/
TEST_F(TemplateRuntimeTest, testReacquireChannel) {
// Create RenderPlayerInfo Directive and wait until PlayerInfo card is displayed.
auto attachmentManager = std::make_shared<StrictMock<MockAttachmentManager>>();
auto avsMessageHeader = std::make_shared<AVSMessageHeader>(PLAYER_INFO.nameSpace, PLAYER_INFO.name, MESSAGE_ID);
std::shared_ptr<AVSDirective> directive =
AVSDirective::create("", avsMessageHeader, PLAYERINFO_PAYLOAD, attachmentManager, "");
EXPECT_CALL(*m_mockGui, renderPlayerInfoCard(PLAYERINFO_PAYLOAD, _, _))
.Times(Exactly(1))
.WillOnce(InvokeWithoutArgs(this, &TemplateRuntimeTest::wakeOnRenderPlayerInfoCard));
AudioPlayerObserverInterface::Context context;
context.audioItemId = AUDIO_ITEM_ID;
context.offset = TIMEOUT;
m_templateRuntime->onPlayerActivityChanged(avsCommon::avs::PlayerActivity::PLAYING, context);
m_templateRuntime->handleDirectiveImmediately(directive);
m_wakeRenderPlayerInfoCardFuture.wait_for(TIMEOUT);
// Send displayCardCleared() to clear card, before setting focus to NONE, send another TemplateCard.
EXPECT_CALL(*m_mockFocusManager, releaseChannel(_, _)).Times(Exactly(1)).WillOnce(InvokeWithoutArgs([this] {
auto releaseChannelSuccess = std::make_shared<std::promise<bool>>();
std::future<bool> returnValue = releaseChannelSuccess->get_future();
releaseChannelSuccess->set_value(true);
wakeOnReleaseChannel();
return returnValue;
}));
m_templateRuntime->displayCardCleared();
m_wakeReleaseChannelFuture.wait_for(TIMEOUT);
// Create RenderTemplate Directive and see if channel is reacquire correctly.
auto avsMessageHeader1 = std::make_shared<AVSMessageHeader>(TEMPLATE.nameSpace, TEMPLATE.name, MESSAGE_ID);
std::shared_ptr<AVSDirective> directive1 =
AVSDirective::create("", avsMessageHeader1, TEMPLATE_PAYLOAD, attachmentManager, "");
EXPECT_CALL(*m_mockGui, renderTemplateCard(TEMPLATE_PAYLOAD, _))
.Times(Exactly(1))
.WillOnce(InvokeWithoutArgs(this, &TemplateRuntimeTest::wakeOnRenderTemplateCard));
m_templateRuntime->handleDirectiveImmediately(directive1);
m_templateRuntime->onFocusChanged(avsCommon::avs::FocusState::NONE);
m_wakeRenderTemplateCardFuture.wait_for(TIMEOUT);
}
} // namespace test
} // namespace templateRuntime
} // namespace capabilityAgents
} // namespace alexaClientSDK