/* * Copyright 2021 The Android Open Source Project * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License 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 "CarTelemetryImpl.h" #include "CarTelemetryInternalImpl.h" #include "FakeLooperWrapper.h" #include "LooperWrapper.h" #include "RingBuffer.h" #include "TelemetryServer.h" #include #include #include #include #include #include #include #include #include #include #include #include // for ::systemTime() #include #include #include #include namespace android { namespace automotive { namespace telemetry { using ::aidl::android::automotive::telemetry::internal::BnCarDataListener; using ::aidl::android::automotive::telemetry::internal::CarDataInternal; using ::aidl::android::automotive::telemetry::internal::ICarTelemetryInternal; using ::aidl::android::frameworks::automotive::telemetry::BnCarTelemetryCallback; using ::aidl::android::frameworks::automotive::telemetry::CallbackConfig; using ::aidl::android::frameworks::automotive::telemetry::CarData; using ::aidl::android::frameworks::automotive::telemetry::ICarTelemetry; using ::ndk::ScopedAStatus; using ::testing::_; using ::testing::ByMove; using ::testing::Return; using ::testing::UnorderedElementsAre; constexpr const std::chrono::nanoseconds kPushCarDataDelayNs = 1000ms; constexpr const std::chrono::nanoseconds kAllowedErrorNs = 100ms; const int kMaxBufferSize = 3; // Because `ScopedAStatus` is move-only, `EXPECT_CALL().WillRepeatedly()` will not work. inline testing::internal::ReturnAction> ReturnOk() { return testing::Return(ByMove(ScopedAStatus::ok())); } // Builds CallbackConfig from clients. CallbackConfig buildConfig(const std::vector& ids) { CallbackConfig config; config.carDataIds = ids; return config; } // Builds incoming CarData from writer clients. CarData buildCarData(int id, const std::vector& content) { CarData msg; msg.id = id; msg.content = content; return msg; } // Builds outgoing CarDataInternal to the CarTelemetryService. CarDataInternal buildCarDataInternal(int id, const std::vector& content) { CarDataInternal msg; msg.id = id; msg.content = content; return msg; } // Mock ICarDataListener, behaves as CarTelemetryService. class MockCarDataListener : public BnCarDataListener { public: MOCK_METHOD(ScopedAStatus, onCarDataReceived, (const std::vector& dataList), (override)); }; // Mock ICarTelemetryCallback, behaves as client application. class MockCarTelemetryCallback : public BnCarTelemetryCallback { public: MOCK_METHOD(ScopedAStatus, onChange, (const std::vector& ids), (override)); }; // The main test class. Tests using `ICarTelemetry` and `ICarTelemetryInternal` interfaces. // Pushing data to the listener is done in the looper - always call `mFakeLooper.poll()`. class TelemetryServerTest : public ::testing::Test { protected: TelemetryServerTest() : mTelemetryServer(&mFakeLooper, kPushCarDataDelayNs, kMaxBufferSize), mDefaultConfig(buildConfig({101})), mMockCarDataListener(ndk::SharedRefBase::make()), mMockCarTelemetryCallback(ndk::SharedRefBase::make()), mTelemetry(ndk::SharedRefBase::make(&mTelemetryServer)), mTelemetryInternal( ndk::SharedRefBase::make(&mTelemetryServer)) {} // Creates an expectation. This is a nice helper that accepts a std::vector, original // EXPECT_CALL() requires creating std::vector variable. testing::internal::TypedExpectation&)>& expectMockListenerToReceive(const std::vector& expected) { return EXPECT_CALL(*mMockCarDataListener, onCarDataReceived(expected)); } void TearDown() override { mTelemetryServer.mCarDataIds.clear(); mTelemetryServer.mCallbacks.clear(); mTelemetryServer.mIdToCallbacksMap.clear(); } FakeLooperWrapper mFakeLooper; TelemetryServer mTelemetryServer; CallbackConfig mDefaultConfig; std::shared_ptr mMockCarDataListener; std::shared_ptr mMockCarTelemetryCallback; std::shared_ptr mTelemetry; std::shared_ptr mTelemetryInternal; }; TEST_F(TelemetryServerTest, WriteReturnsOk) { std::vector dataList = {buildCarData(101, {1})}; auto status = mTelemetry->write(dataList); EXPECT_TRUE(status.isOk()) << status.getMessage(); } TEST_F(TelemetryServerTest, AddCarDataIdsReturnsOk) { auto status = mTelemetryInternal->addCarDataIds({101}); EXPECT_TRUE(status.isOk()) << status.getMessage(); } TEST_F(TelemetryServerTest, AddCarDataIdsNotifiesInterestedCallbacks) { CallbackConfig config = buildConfig({101, 102}); std::shared_ptr mockCallback = ndk::SharedRefBase::make(); mTelemetry->addCallback(config, mockCallback); // mDefaultConfig only contains ID 101 mTelemetry->addCallback(mDefaultConfig, mMockCarTelemetryCallback); EXPECT_CALL(*mMockCarTelemetryCallback, onChange(UnorderedElementsAre(101))) .Times(1) .WillOnce(ReturnOk()); EXPECT_CALL(*mockCallback, onChange(UnorderedElementsAre(101, 102))) .Times(1) .WillOnce(ReturnOk()); mTelemetryInternal->addCarDataIds({101, 102, 103, 104}); } TEST_F(TelemetryServerTest, RemoveCarDataIdsReturnsOk) { mTelemetryInternal->addCarDataIds({101, 102, 103}); EXPECT_EQ(3, mTelemetryServer.mCarDataIds.size()); auto status = mTelemetryInternal->removeCarDataIds({101, 103}); EXPECT_TRUE(status.isOk()) << status.getMessage(); EXPECT_EQ(1, mTelemetryServer.mCarDataIds.size()); EXPECT_NE(mTelemetryServer.mCarDataIds.end(), mTelemetryServer.mCarDataIds.find(102)); } TEST_F(TelemetryServerTest, RemoveCarDataIdsNotifiesInterestedCallbacks) { // should only receive updates on IDs 101, 102, 103 CallbackConfig config = buildConfig({101, 102, 103}); mTelemetry->addCallback(config, mMockCarTelemetryCallback); EXPECT_CALL(*mMockCarTelemetryCallback, onChange(UnorderedElementsAre(101, 102, 103))) .Times(1) .WillOnce(ReturnOk()); EXPECT_CALL(*mMockCarTelemetryCallback, onChange(UnorderedElementsAre(101, 102))) .Times(1) .WillOnce(ReturnOk()); mTelemetryInternal->addCarDataIds({101, 102, 103, 104}); mTelemetryInternal->removeCarDataIds({103, 104}); } TEST_F(TelemetryServerTest, SetListenerReturnsOk) { auto status = mTelemetryInternal->setListener(mMockCarDataListener); EXPECT_TRUE(status.isOk()) << status.getMessage(); } TEST_F(TelemetryServerTest, SetListenerAllowedWhenAlreadySubscribed) { mTelemetryInternal->setListener(mMockCarDataListener); auto status = mTelemetryInternal->setListener(ndk::SharedRefBase::make()); EXPECT_TRUE(status.isOk()) << status.getMessage(); } TEST_F(TelemetryServerTest, ClearListenerWorks) { mTelemetryInternal->setListener(mMockCarDataListener); mTelemetryInternal->clearListener(); auto status = mTelemetryInternal->setListener(mMockCarDataListener); EXPECT_TRUE(status.isOk()) << status.getMessage(); } TEST_F(TelemetryServerTest, ClearListenerRemovesPushMessagesFromLooper) { std::vector dataList = {buildCarData(101, {1})}; mTelemetry->write(dataList); mTelemetryInternal->setListener(mMockCarDataListener); EXPECT_NE(mFakeLooper.getNextMessageUptime(), FakeLooperWrapper::kNoScheduledMessage); mTelemetryInternal->clearListener(); EXPECT_EQ(mFakeLooper.getNextMessageUptime(), FakeLooperWrapper::kNoScheduledMessage); } TEST_F(TelemetryServerTest, AddCallbackReturnsOk) { auto status = mTelemetry->addCallback(mDefaultConfig, mMockCarTelemetryCallback); EXPECT_TRUE(status.isOk()) << status.getMessage(); } TEST_F(TelemetryServerTest, AddCallbackReturnsErrorForExistingCallback) { mTelemetry->addCallback(mDefaultConfig, mMockCarTelemetryCallback); auto status = mTelemetry->addCallback(mDefaultConfig, mMockCarTelemetryCallback); EXPECT_FALSE(status.isOk()) << status.getMessage(); } TEST_F(TelemetryServerTest, AddCallbackReceivesCarDataIds) { CallbackConfig config = buildConfig({101, 102, 103}); mTelemetryInternal->addCarDataIds({101, 102, 103, 104}); EXPECT_CALL(*mMockCarTelemetryCallback, onChange(UnorderedElementsAre(101, 102, 103))) .Times(1) .WillOnce(ReturnOk()); auto status = mTelemetry->addCallback(config, mMockCarTelemetryCallback); } TEST_F(TelemetryServerTest, RemoveCallbackReturnsOk) { mTelemetry->addCallback(mDefaultConfig, mMockCarTelemetryCallback); auto status = mTelemetry->removeCallback(mMockCarTelemetryCallback); EXPECT_TRUE(status.isOk()) << status.getMessage(); EXPECT_EQ(0, mTelemetryServer.mCallbacks.size()); EXPECT_EQ(0, mTelemetryServer.mIdToCallbacksMap.size()); } TEST_F(TelemetryServerTest, RemoveCallbackReturnsErrorForNonexistentCallback) { auto status = mTelemetry->removeCallback(mMockCarTelemetryCallback); EXPECT_FALSE(status.isOk()) << status.getMessage(); } TEST_F(TelemetryServerTest, WriteSchedulesNextMessageAfterRightDelay) { std::vector dataList = {buildCarData(101, {1})}; mTelemetryInternal->setListener(mMockCarDataListener); mTelemetry->write(dataList); EXPECT_NEAR(mFakeLooper.getNextMessageUptime(), ::systemTime() + kPushCarDataDelayNs.count(), kAllowedErrorNs.count()); } TEST_F(TelemetryServerTest, SetListenerSchedulesNextMessageAfterRightDelay) { std::vector dataList = {buildCarData(101, {1})}; mTelemetry->write(dataList); mTelemetryInternal->setListener(mMockCarDataListener); EXPECT_NEAR(mFakeLooper.getNextMessageUptime(), ::systemTime() + kPushCarDataDelayNs.count(), kAllowedErrorNs.count()); } TEST_F(TelemetryServerTest, BuffersOnlyLimitedData) { mTelemetryInternal->setListener(mMockCarDataListener); std::vector dataList1 = {buildCarData(10, {1, 2}), buildCarData(11, {2, 3})}; std::vector dataList2 = {buildCarData(101, {1, 2}), buildCarData(102, {2, 3}), buildCarData(103, {3, 4}), buildCarData(104, {4, 5})}; mTelemetryInternal->addCarDataIds({10, 11, 101, 102, 103, 104}); mTelemetry->write(dataList1); mTelemetry->write(dataList2); // Only the last 3 CarData should be received, because kMaxBufferSize = 3. expectMockListenerToReceive({buildCarDataInternal(102, {2, 3})}).WillOnce(ReturnOk()); expectMockListenerToReceive({buildCarDataInternal(103, {3, 4})}).WillOnce(ReturnOk()); expectMockListenerToReceive({buildCarDataInternal(104, {4, 5})}).WillOnce(ReturnOk()); mFakeLooper.poll(); mFakeLooper.poll(); mFakeLooper.poll(); mFakeLooper.poll(); } // Data is filtered out when mTelemetryInternal->addCarDataIds() is not called TEST_F(TelemetryServerTest, WriteFiltersDataBasedOnId) { std::vector dataList = {buildCarData(101, {1})}; mTelemetryInternal->setListener(mMockCarDataListener); mTelemetry->write(dataList); expectMockListenerToReceive({buildCarDataInternal(101, {1})}).Times(0); mFakeLooper.poll(); } // First sets the listener, then writes CarData. TEST_F(TelemetryServerTest, WhenListenerIsAlreadySetItPushesData) { std::vector dataList = {buildCarData(101, {1})}; mTelemetryInternal->addCarDataIds({101}); mTelemetryInternal->setListener(mMockCarDataListener); mTelemetry->write(dataList); expectMockListenerToReceive({buildCarDataInternal(101, {1})}).Times(1).WillOnce(ReturnOk()); mFakeLooper.poll(); } // First writes CarData, only then sets the listener. TEST_F(TelemetryServerTest, WhenListenerIsSetLaterItPushesData) { std::vector dataList = {buildCarData(101, {1})}; mTelemetryInternal->addCarDataIds({101}); mTelemetry->write(dataList); mTelemetryInternal->setListener(mMockCarDataListener); expectMockListenerToReceive({buildCarDataInternal(101, {1})}).Times(1).WillOnce(ReturnOk()); mFakeLooper.poll(); } TEST_F(TelemetryServerTest, WriteDuringPushingDataToListener) { mTelemetryInternal->addCarDataIds({101, 102, 103}); std::vector dataList = {buildCarData(101, {1}), buildCarData(102, {1})}; std::vector dataList2 = {buildCarData(103, {1})}; mTelemetryInternal->setListener(mMockCarDataListener); mTelemetry->write(dataList); expectMockListenerToReceive({buildCarDataInternal(101, {1})}).WillOnce(ReturnOk()); expectMockListenerToReceive({buildCarDataInternal(102, {1})}).WillOnce(ReturnOk()); expectMockListenerToReceive({buildCarDataInternal(103, {1})}).WillOnce(ReturnOk()); mFakeLooper.poll(); // sends only 1 CarData (or possibly 2 depenending on impl) mTelemetry->write(dataList2); mFakeLooper.poll(); // all the polls below send the rest of the CarData mFakeLooper.poll(); mFakeLooper.poll(); // extra poll to verify there was not excess push calls } TEST_F(TelemetryServerTest, ClearListenerDuringPushingDataToListener) { std::vector dataList = {buildCarData(101, {1})}; mTelemetryInternal->addCarDataIds({101}); mTelemetryInternal->setListener(mMockCarDataListener); mTelemetry->write(dataList); expectMockListenerToReceive({buildCarDataInternal(101, {1})}).Times(1).WillOnce(ReturnOk()); mFakeLooper.poll(); mTelemetry->write(dataList); mTelemetryInternal->clearListener(); mFakeLooper.poll(); } TEST_F(TelemetryServerTest, RetriesPushAgainIfListenerFails) { std::vector dataList = {buildCarData(101, {1})}; mTelemetryInternal->addCarDataIds({101}); mTelemetryInternal->setListener(mMockCarDataListener); mTelemetry->write(dataList); expectMockListenerToReceive({buildCarDataInternal(101, {1})}) .WillOnce(Return(ByMove(ScopedAStatus::fromExceptionCode(::EX_TRANSACTION_FAILED)))) .WillOnce(ReturnOk()); mFakeLooper.poll(); // listener returns ::EX_TRANSACTION_FAILED mFakeLooper.poll(); } // Tests a corner case to make sure `TelemetryServer::mPendingCarDataInternals` variable // is handled properly when transaction fails and clearListener() is called. TEST_F(TelemetryServerTest, ClearListenerDuringPushingDataAndSetListenerAgain) { std::vector dataList = {buildCarData(101, {1})}; mTelemetryInternal->addCarDataIds({101}); mTelemetryInternal->setListener(mMockCarDataListener); mTelemetry->write(dataList); expectMockListenerToReceive({buildCarDataInternal(101, {1})}) .WillOnce(Return(ByMove(ScopedAStatus::fromExceptionCode(::EX_TRANSACTION_FAILED)))) .WillOnce(ReturnOk()); mFakeLooper.poll(); // listener returns ::EX_TRANSACTION_FAILED mTelemetryInternal->clearListener(); mFakeLooper.poll(); // nothing happens mTelemetryInternal->setListener(mMockCarDataListener); mFakeLooper.poll(); // should work } // Directly calls pushCarDataToListeners() to make sure it can handle edge-cases. TEST_F(TelemetryServerTest, NoListenerButMultiplePushes) { std::vector dataList = {buildCarData(101, {1})}; mTelemetryInternal->addCarDataIds({101}); mTelemetry->write(dataList); mTelemetryServer.pushCarDataToListeners(); mTelemetryServer.pushCarDataToListeners(); mTelemetryServer.pushCarDataToListeners(); EXPECT_CALL(*mMockCarDataListener, onCarDataReceived(_)).Times(0); } // Directly calls pushCarDataToListeners() to make sure it can handle edge-cases. TEST_F(TelemetryServerTest, NoDataButMultiplePushes) { mTelemetryInternal->setListener(mMockCarDataListener); mTelemetryServer.pushCarDataToListeners(); mTelemetryServer.pushCarDataToListeners(); mTelemetryServer.pushCarDataToListeners(); EXPECT_CALL(*mMockCarDataListener, onCarDataReceived(_)).Times(0); } } // namespace telemetry } // namespace automotive } // namespace android