blob: ca5c2d5a2e176ff9863bd24eb985212ad04f0cca [file] [log] [blame]
// Copyright (C) 2023 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 "GRPCVehicleHardware.h"
#include "GRPCVehicleProxyServer.h"
#include "IVehicleHardware.h"
#include "VehicleServer.grpc.pb.h"
#include "VehicleServer.pb.h"
#include <gmock/gmock.h>
#include <grpc++/grpc++.h>
#include <gtest/gtest.h>
#include <chrono>
#include <memory>
#include <string>
#include <thread>
#include <utility>
namespace android::hardware::automotive::vehicle::virtualization {
namespace aidlvhal = ::aidl::android::hardware::automotive::vehicle;
using ::testing::_;
using ::testing::DoAll;
using ::testing::Return;
using ::testing::SaveArg;
const std::string kFakeServerAddr = "0.0.0.0:54321";
class VehicleHardwareForTest : public IVehicleHardware {
public:
void registerOnPropertyChangeEvent(
std::unique_ptr<const PropertyChangeCallback> callback) override {
mOnProp = std::move(callback);
}
void onPropertyEvent(std::vector<aidlvhal::VehiclePropValue> values) {
if (mOnProp) {
(*mOnProp)(std::move(values));
}
}
// Functions that we do not care.
std::vector<aidlvhal::VehiclePropConfig> getAllPropertyConfigs() const override { return {}; }
aidlvhal::StatusCode setValues(
std::shared_ptr<const SetValuesCallback> callback,
const std::vector<aidlvhal::SetValueRequest>& requests) override {
return aidlvhal::StatusCode::OK;
}
aidlvhal::StatusCode getValues(
std::shared_ptr<const GetValuesCallback> callback,
const std::vector<aidlvhal::GetValueRequest>& requests) const override {
return aidlvhal::StatusCode::OK;
}
DumpResult dump(const std::vector<std::string>& options) override { return {}; }
aidlvhal::StatusCode checkHealth() override { return aidlvhal::StatusCode::OK; }
void registerOnPropertySetErrorEvent(
std::unique_ptr<const PropertySetErrorCallback> callback) override {}
private:
std::unique_ptr<const PropertyChangeCallback> mOnProp;
};
class MockVehicleHardware : public IVehicleHardware {
public:
// Mock methods from IVehicleHardware
MOCK_METHOD(std::vector<aidlvhal::VehiclePropConfig>, getAllPropertyConfigs, (),
(const, override));
MOCK_METHOD((aidlvhal::StatusCode), setValues,
(std::shared_ptr<const SetValuesCallback> callback,
const std::vector<aidlvhal::SetValueRequest>& requests),
(override));
MOCK_METHOD((aidlvhal::StatusCode), getValues,
(std::shared_ptr<const GetValuesCallback> callback,
const std::vector<aidlvhal::GetValueRequest>& requests),
(const, override));
MOCK_METHOD(DumpResult, dump, (const std::vector<std::string>& options), (override));
MOCK_METHOD(aidlvhal::StatusCode, checkHealth, (), (override));
MOCK_METHOD(void, registerOnPropertyChangeEvent,
(std::unique_ptr<const PropertyChangeCallback> callback), (override));
MOCK_METHOD(void, registerOnPropertySetErrorEvent,
(std::unique_ptr<const PropertySetErrorCallback> callback), (override));
MOCK_METHOD(std::chrono::nanoseconds, getPropertyOnChangeEventBatchingWindow, (), (override));
MOCK_METHOD(aidlvhal::StatusCode, subscribe, (aidlvhal::SubscribeOptions options), (override));
MOCK_METHOD(aidlvhal::StatusCode, unsubscribe, (int32_t propId, int32_t areaId), (override));
MOCK_METHOD(aidlvhal::StatusCode, updateSampleRate,
(int32_t propId, int32_t areaId, float sampleRate), (override));
};
TEST(GRPCVehicleProxyServerUnitTest, ClientConnectDisconnect) {
auto testHardware = std::make_unique<VehicleHardwareForTest>();
// HACK: manipulate the underlying hardware via raw pointer for testing.
auto* testHardwareRaw = testHardware.get();
auto vehicleServer =
std::make_unique<GrpcVehicleProxyServer>(kFakeServerAddr, std::move(testHardware));
vehicleServer->Start();
constexpr auto kWaitForConnectionMaxTime = std::chrono::seconds(5);
constexpr auto kWaitForStreamStartTime = std::chrono::seconds(1);
constexpr auto kWaitForUpdateDeliveryTime = std::chrono::milliseconds(100);
auto updateReceived1 = std::make_shared<bool>(false);
auto vehicleHardware1 = std::make_unique<GRPCVehicleHardware>(kFakeServerAddr);
vehicleHardware1->registerOnPropertyChangeEvent(
std::make_unique<const IVehicleHardware::PropertyChangeCallback>(
[updateReceived1](const auto&) { *updateReceived1 = true; }));
EXPECT_TRUE(vehicleHardware1->waitForConnected(kWaitForConnectionMaxTime));
std::this_thread::sleep_for(kWaitForStreamStartTime);
// Client hardware 1 received update from the server.
EXPECT_FALSE(*updateReceived1);
testHardwareRaw->onPropertyEvent({});
// Wait for the update delivery.
std::this_thread::sleep_for(kWaitForUpdateDeliveryTime);
EXPECT_TRUE(*updateReceived1);
// Reset.
*updateReceived1 = false;
auto updateReceived2 = std::make_shared<bool>(false);
auto vehicleHardware2 = std::make_unique<GRPCVehicleHardware>(kFakeServerAddr);
vehicleHardware2->registerOnPropertyChangeEvent(
std::make_unique<const IVehicleHardware::PropertyChangeCallback>(
[updateReceived2](const auto&) { *updateReceived2 = true; }));
EXPECT_TRUE(vehicleHardware2->waitForConnected(kWaitForConnectionMaxTime));
std::this_thread::sleep_for(kWaitForStreamStartTime);
// Both client hardware 1 and 2 received update from the server.
EXPECT_FALSE(*updateReceived1);
EXPECT_FALSE(*updateReceived2);
testHardwareRaw->onPropertyEvent({});
// Wait for the update delivery.
std::this_thread::sleep_for(kWaitForUpdateDeliveryTime);
EXPECT_TRUE(*updateReceived1);
EXPECT_TRUE(*updateReceived2);
// Reset.
*updateReceived1 = false;
*updateReceived2 = false;
vehicleHardware1.reset();
// Client 1 exited, only client hardware 2 received update from the server.
EXPECT_FALSE(*updateReceived1);
EXPECT_FALSE(*updateReceived2);
testHardwareRaw->onPropertyEvent({});
// Wait for the update delivery.
std::this_thread::sleep_for(kWaitForUpdateDeliveryTime);
EXPECT_FALSE(*updateReceived1);
EXPECT_TRUE(*updateReceived2);
vehicleServer->Shutdown().Wait();
}
TEST(GRPCVehicleProxyServerUnitTest, Subscribe) {
auto mockHardware = std::make_unique<MockVehicleHardware>();
// We make sure this is alive inside the function scope.
MockVehicleHardware* mockHardwarePtr = mockHardware.get();
GrpcVehicleProxyServer server = GrpcVehicleProxyServer("", std::move(mockHardware));
::grpc::ServerContext context;
proto::SubscribeRequest request;
proto::VehicleHalCallStatus returnStatus;
aidlvhal::SubscribeOptions aidlOptions;
request.mutable_options()->set_prop_id(1);
request.mutable_options()->add_area_ids(2);
request.mutable_options()->set_sample_rate(1.234);
request.mutable_options()->set_resolution(0.01);
request.mutable_options()->set_enable_variable_update_rate(true);
EXPECT_CALL(*mockHardwarePtr, subscribe(_))
.WillOnce(DoAll(SaveArg<0>(&aidlOptions), Return(aidlvhal::StatusCode::OK)));
auto grpcStatus = server.Subscribe(&context, &request, &returnStatus);
EXPECT_TRUE(grpcStatus.ok());
EXPECT_EQ(returnStatus.status_code(), proto::StatusCode::OK);
EXPECT_EQ(aidlOptions.propId, 1);
EXPECT_EQ(aidlOptions.areaIds, std::vector<int32_t>{2});
EXPECT_FLOAT_EQ(aidlOptions.sampleRate, 1.234);
EXPECT_FLOAT_EQ(aidlOptions.resolution, 0.01);
EXPECT_TRUE(aidlOptions.enableVariableUpdateRate);
}
TEST(GRPCVehicleProxyServerUnitTest, SubscribeNotAvailable) {
auto mockHardware = std::make_unique<MockVehicleHardware>();
// We make sure this is alive inside the function scope.
MockVehicleHardware* mockHardwarePtr = mockHardware.get();
GrpcVehicleProxyServer server = GrpcVehicleProxyServer("", std::move(mockHardware));
::grpc::ServerContext context;
proto::SubscribeRequest request;
proto::VehicleHalCallStatus returnStatus;
EXPECT_CALL(*mockHardwarePtr, subscribe(_))
.WillOnce(Return(aidlvhal::StatusCode::NOT_AVAILABLE));
auto grpcStatus = server.Subscribe(&context, &request, &returnStatus);
EXPECT_TRUE(grpcStatus.ok());
EXPECT_EQ(returnStatus.status_code(), proto::StatusCode::NOT_AVAILABLE);
}
TEST(GRPCVehicleProxyServerUnitTest, Unsubscribe) {
auto mockHardware = std::make_unique<MockVehicleHardware>();
// We make sure this is alive inside the function scope.
MockVehicleHardware* mockHardwarePtr = mockHardware.get();
GrpcVehicleProxyServer server = GrpcVehicleProxyServer("", std::move(mockHardware));
::grpc::ServerContext context;
proto::UnsubscribeRequest request;
proto::VehicleHalCallStatus returnStatus;
request.set_prop_id(1);
request.set_area_id(2);
EXPECT_CALL(*mockHardwarePtr, unsubscribe(1, 2)).WillOnce(Return(aidlvhal::StatusCode::OK));
auto grpcStatus = server.Unsubscribe(&context, &request, &returnStatus);
EXPECT_TRUE(grpcStatus.ok());
EXPECT_EQ(returnStatus.status_code(), proto::StatusCode::OK);
}
} // namespace android::hardware::automotive::vehicle::virtualization