From 67e95ae9fe637cb90d1d53a98471dd39992bb38c Mon Sep 17 00:00:00 2001 From: Gianfranco Berardi <54074967+gberardi-pillar@users.noreply.github.com> Date: Wed, 28 Feb 2024 15:10:07 +0000 Subject: [PATCH] Validate a TxDefaultProfile. These changes apply to K01FR52 and K01FR53. Signed-off-by: Christoph <367712+folkengine@users.noreply.github.com> Signed-off-by: Gianfranco Berardi <54074967+gberardi-pillar@users.noreply.github.com> --- include/ocpp/v201/evse.hpp | 11 ++ include/ocpp/v201/smart_charging.hpp | 21 +- lib/ocpp/v201/evse.cpp | 9 + lib/ocpp/v201/smart_charging.cpp | 50 ++++- .../ocpp/v201/test_smart_charging_handler.cpp | 187 ++++++++++++------ 5 files changed, 212 insertions(+), 66 deletions(-) diff --git a/include/ocpp/v201/evse.hpp b/include/ocpp/v201/evse.hpp index 259e71ebb7..9df802c30a 100644 --- a/include/ocpp/v201/evse.hpp +++ b/include/ocpp/v201/evse.hpp @@ -71,6 +71,17 @@ class Evse { /// \return uint32_t get_number_of_connectors(); + /// \brief Returns true if evse_id is 0. + /// From 1.51.1 SetChargingProfileRequest: "TxDefaultProfile an evseId=0 applies the profile to each individual + /// evse. For ChargingStationMaxProfile and ChargingStationExternalConstraints an evseId=0 contains an overal limit + /// for the whole Charging Station." + /// \return + bool is_station_wide() const; + + /// \brief Returns true if evse_id is 0. + /// \return + static bool is_station_wide_id(int32_t id); + /// \brief Opens a new transaction /// \param transaction_id id of the transaction /// \param connector_id id of the connector diff --git a/include/ocpp/v201/smart_charging.hpp b/include/ocpp/v201/smart_charging.hpp index fadf6016a0..ee6e53116f 100644 --- a/include/ocpp/v201/smart_charging.hpp +++ b/include/ocpp/v201/smart_charging.hpp @@ -27,6 +27,12 @@ enum class ProfileValidationResultEnum { ChargingProfileExtraneousStartSchedule, ChargingSchedulePeriodsOutOfOrder, ChargingSchedulePeriodInvalidPhaseToUse, + DuplicateTxDefaultProfileFound +}; + +struct EvseProfile { + int32_t evse_id; + ChargingProfile profile; }; /// \brief This class handles and maintains incoming ChargingProfiles and contains the logic @@ -35,11 +41,16 @@ class SmartChargingHandler { private: std::shared_ptr database_handler; // cppcheck-suppress unusedStructMember - std::vector charging_profiles; + std::vector charging_profiles; public: explicit SmartChargingHandler(); + /// + /// \brief validates the given \p profile according to the specification + /// + ProfileValidationResultEnum validate_tx_default_profile(const ChargingProfile& profile, Evse& evse) const; + /// /// \brief validates the given \p profile according to the specification /// @@ -48,8 +59,14 @@ class SmartChargingHandler { /// \brief validates that the given \p profile has valid charging schedules ProfileValidationResultEnum validate_profile_schedules(const ChargingProfile& profile) const; + /// /// \brief Adds a given \p profile to our stored list of profiles - void add_profile(const ChargingProfile& profile); + /// + void add_profile(int32_t evse_id, ChargingProfile& profile); + +private: + std::vector get_evse_specific_tx_default_profiles() const; + std::vector get_station_wide_tx_default_profiles() const; }; } // namespace ocpp::v201 diff --git a/lib/ocpp/v201/evse.cpp b/lib/ocpp/v201/evse.cpp index 5eedfb9690..91cb4de2c4 100644 --- a/lib/ocpp/v201/evse.cpp +++ b/lib/ocpp/v201/evse.cpp @@ -62,6 +62,15 @@ EVSE Evse::get_evse_info() { return evse; } +bool Evse::is_station_wide() const { + return is_station_wide_id(evse_id); +} + +bool Evse::is_station_wide_id(int32_t id) { + const int32_t STATION_WIDE_ID = 0; + return id == STATION_WIDE_ID; +} + uint32_t Evse::get_number_of_connectors() { return static_cast(this->id_connector_map.size()); } diff --git a/lib/ocpp/v201/smart_charging.cpp b/lib/ocpp/v201/smart_charging.cpp index d28e60d726..773f5b1f88 100644 --- a/lib/ocpp/v201/smart_charging.cpp +++ b/lib/ocpp/v201/smart_charging.cpp @@ -4,8 +4,10 @@ #include "everest/logging.hpp" #include "ocpp/common/types.hpp" #include "ocpp/v201/enums.hpp" +#include "ocpp/v201/evse.hpp" #include "ocpp/v201/ocpp_types.hpp" #include "ocpp/v201/transaction.hpp" +#include #include #include @@ -16,6 +18,21 @@ namespace ocpp::v201 { SmartChargingHandler::SmartChargingHandler() { } +ProfileValidationResultEnum SmartChargingHandler::validate_tx_default_profile(const ChargingProfile& profile, + Evse& evse) const { + auto profiles = + evse.is_station_wide() ? get_evse_specific_tx_default_profiles() : get_station_wide_tx_default_profiles(); + for (auto iter = profiles.begin(); iter != profiles.end(); ++iter) { + if (iter->profile.stackLevel == profile.stackLevel) { + if (iter->profile.id != profile.id) { + return ProfileValidationResultEnum::DuplicateTxDefaultProfileFound; + } + } + } + + return ProfileValidationResultEnum::Valid; +} + ProfileValidationResultEnum SmartChargingHandler::validate_tx_profile(const ChargingProfile& profile, Evse& evse) const { if (!profile.transactionId.has_value()) { @@ -36,8 +53,9 @@ ProfileValidationResultEnum SmartChargingHandler::validate_tx_profile(const Char return ProfileValidationResultEnum::TxProfileTransactionNotOnEvse; } - auto conflicts_with = [&profile](const ChargingProfile& candidate) { - return candidate.transactionId == profile.transactionId && candidate.stackLevel == profile.stackLevel; + auto conflicts_with = [&profile](const EvseProfile& candidate) { + return candidate.profile.transactionId == profile.transactionId && + candidate.profile.stackLevel == profile.stackLevel; }; if (std::any_of(charging_profiles.begin(), charging_profiles.end(), conflicts_with)) { return ProfileValidationResultEnum::TxProfileConflictingStackLevel; @@ -97,8 +115,32 @@ ProfileValidationResultEnum SmartChargingHandler::validate_profile_schedules(con return ProfileValidationResultEnum::Valid; } -void SmartChargingHandler::add_profile(const ChargingProfile& profile) { - charging_profiles.push_back(profile); +void SmartChargingHandler::add_profile(int32_t evse_id, ChargingProfile& profile) { + charging_profiles.push_back(EvseProfile{evse_id = evse_id, profile = profile}); +} + +std::vector SmartChargingHandler::get_evse_specific_tx_default_profiles() const { + std::vector evse_specific_tx_default_profiles; + auto pred = [](const EvseProfile& candidate) { + return !Evse::is_station_wide_id(candidate.evse_id) && + candidate.profile.chargingProfilePurpose == ChargingProfilePurposeEnum::TxDefaultProfile; + }; + std::copy_if(charging_profiles.begin(), charging_profiles.end(), + std::back_inserter(evse_specific_tx_default_profiles), pred); + + return evse_specific_tx_default_profiles; +} + +std::vector SmartChargingHandler::get_station_wide_tx_default_profiles() const { + std::vector station_wide_tx_default_profiles; + auto pred = [](const EvseProfile& candidate) { + return Evse::is_station_wide_id(candidate.evse_id) && + candidate.profile.chargingProfilePurpose == ChargingProfilePurposeEnum::TxDefaultProfile; + }; + std::copy_if(charging_profiles.begin(), charging_profiles.end(), + std::back_inserter(station_wide_tx_default_profiles), pred); + + return station_wide_tx_default_profiles; } } // namespace ocpp::v201 diff --git a/tests/lib/ocpp/v201/test_smart_charging_handler.cpp b/tests/lib/ocpp/v201/test_smart_charging_handler.cpp index f804083d91..ec9a5c81b5 100644 --- a/tests/lib/ocpp/v201/test_smart_charging_handler.cpp +++ b/tests/lib/ocpp/v201/test_smart_charging_handler.cpp @@ -23,7 +23,13 @@ namespace ocpp::v201 { -class ChargepointTestFixtureV201 : public testing::Test { +static const int STATION_WIDE_ID = 0; +static const int DEFAULT_EVSE_ID = 1; +static const int DEFAULT_PROFILE_ID = 1; +static const int DEFAULT_STACK_LEVEL = 1; + +class ChargepointTestFixtureV201 + : public ::testing::TestWithParam> { protected: void SetUp() override { } @@ -99,10 +105,10 @@ class ChargepointTestFixtureV201 : public testing::Test { } ChargingProfile - create_tx_profile(ChargingSchedule charging_schedule, std::string transaction_id, int stack_level = 1, - ChargingProfileKindEnum charging_profile_kind = ChargingProfileKindEnum::Absolute) { - auto charging_profile_id = 1; - auto charging_profile_purpose = ChargingProfilePurposeEnum::TxProfile; + create_charging_profile(int32_t charging_profile_id, ChargingProfilePurposeEnum charging_profile_purpose, + ChargingSchedule charging_schedule, std::string transaction_id, + ChargingProfileKindEnum charging_profile_kind = ChargingProfileKindEnum::Absolute, + int stack_level = DEFAULT_STACK_LEVEL) { auto recurrency_kind = RecurrencyKindEnum::Daily; std::vector charging_schedules = {charging_schedule}; return ChargingProfile{.id = charging_profile_id, @@ -118,8 +124,8 @@ class ChargepointTestFixtureV201 : public testing::Test { } ChargingProfile create_tx_profile_with_missing_transaction_id(ChargingSchedule charging_schedule) { - auto charging_profile_id = 1; - auto stack_level = 1; + auto charging_profile_id = DEFAULT_PROFILE_ID; + auto stack_level = DEFAULT_STACK_LEVEL; auto charging_profile_purpose = ChargingProfilePurposeEnum::TxProfile; auto charging_profile_kind = ChargingProfileKindEnum::Absolute; auto recurrency_kind = RecurrencyKindEnum::Daily; @@ -176,42 +182,47 @@ class ChargepointTestFixtureV201 : public testing::Test { std::chrono::seconds(static_cast(1)), std::chrono::seconds(static_cast(1))); } + void install_profile_on_evse(int evse_id, int profile_id) { + create_evse_with_id(evse_id); + auto existing_profile = create_charging_profile(profile_id, ChargingProfilePurposeEnum::TxDefaultProfile, + create_charge_schedule(ChargingRateUnitEnum::A), uuid()); + handler.add_profile(evse_id, existing_profile); + } + // Default values used within the tests std::map> evses; std::shared_ptr database_handler; - const int evse_id = 1; bool ignore_no_transaction = true; - const int profile_max_stack_level = 1; - const int max_charging_profiles_installed = 1; - const int charging_schedule_max_periods = 1; DeviceModel device_model = create_device_model(); SmartChargingHandler handler = create_smart_charging_handler(); boost::uuids::random_generator uuid_generator = boost::uuids::random_generator(); }; TEST_F(ChargepointTestFixtureV201, K01FR03_IfTxProfileIsMissingTransactionId_ThenProfileIsInvalid) { - create_evse_with_id(evse_id); + create_evse_with_id(DEFAULT_EVSE_ID); auto profile = create_tx_profile_with_missing_transaction_id(create_charge_schedule(ChargingRateUnitEnum::A)); - auto sut = handler.validate_tx_profile(profile, *evses[evse_id]); + auto sut = handler.validate_tx_profile(profile, *evses[DEFAULT_EVSE_ID]); EXPECT_THAT(sut, testing::Eq(ProfileValidationResultEnum::TxProfileMissingTransactionId)); } TEST_F(ChargepointTestFixtureV201, K01FR16_IfTxProfileHasEvseIdNotGreaterThanZero_ThenProfileIsInvalid) { - auto wrong_evse_id = 0; - create_evse_with_id(0); - auto profile = create_tx_profile(create_charge_schedule(ChargingRateUnitEnum::A), uuid()); + auto wrong_evse_id = STATION_WIDE_ID; + create_evse_with_id(wrong_evse_id); + auto profile = create_charging_profile(DEFAULT_PROFILE_ID, ChargingProfilePurposeEnum::TxProfile, + create_charge_schedule(ChargingRateUnitEnum::A), uuid()); auto sut = handler.validate_tx_profile(profile, *evses[wrong_evse_id]); EXPECT_THAT(sut, testing::Eq(ProfileValidationResultEnum::TxProfileEvseIdNotGreaterThanZero)); } TEST_F(ChargepointTestFixtureV201, K01FR33_IfTxProfileTransactionIsNotOnEvse_ThenProfileIsInvalid) { - create_evse_with_id(evse_id); - open_evse_transaction(evse_id, "wrong transaction id"); - auto profile = create_tx_profile(create_charge_schedule(ChargingRateUnitEnum::A), uuid()); - auto sut = handler.validate_tx_profile(profile, *evses[evse_id]); + create_evse_with_id(DEFAULT_EVSE_ID); + open_evse_transaction(DEFAULT_EVSE_ID, "wrong transaction id"); + auto profile = create_charging_profile(DEFAULT_PROFILE_ID, ChargingProfilePurposeEnum::TxProfile, + create_charge_schedule(ChargingRateUnitEnum::A), uuid()); + auto sut = handler.validate_tx_profile(profile, *evses[DEFAULT_EVSE_ID]); EXPECT_THAT(sut, testing::Eq(ProfileValidationResultEnum::TxProfileTransactionNotOnEvse)); } @@ -221,71 +232,81 @@ TEST_F(ChargepointTestFixtureV201, K01FR09_IfTxProfileEvseHasNoActiveTransaction auto meter_start = MeterValue(); auto id_token = IdToken(); auto date_time = ocpp::DateTime("2024-01-17T17:00:00"); - create_evse_with_id(evse_id); - auto profile = create_tx_profile(create_charge_schedule(ChargingRateUnitEnum::A), uuid()); - auto sut = handler.validate_tx_profile(profile, *evses[evse_id]); + create_evse_with_id(DEFAULT_EVSE_ID); + auto profile = create_charging_profile(DEFAULT_PROFILE_ID, ChargingProfilePurposeEnum::TxProfile, + create_charge_schedule(ChargingRateUnitEnum::A), uuid()); + auto sut = handler.validate_tx_profile(profile, *evses[DEFAULT_EVSE_ID]); EXPECT_THAT(sut, testing::Eq(ProfileValidationResultEnum::TxProfileEvseHasNoActiveTransaction)); } TEST_F(ChargepointTestFixtureV201, K01FR06_IfTxProfileHasSameTransactionAndStackLevelAsAnotherTxProfile_ThenProfileIsInvalid) { - create_evse_with_id(evse_id); + create_evse_with_id(DEFAULT_EVSE_ID); std::string transaction_id = uuid(); - open_evse_transaction(evse_id, transaction_id); + open_evse_transaction(DEFAULT_EVSE_ID, transaction_id); auto same_stack_level = 42; - auto profile_1 = - create_tx_profile(create_charge_schedule(ChargingRateUnitEnum::A), transaction_id, same_stack_level); - auto profile_2 = - create_tx_profile(create_charge_schedule(ChargingRateUnitEnum::A), transaction_id, same_stack_level); - handler.add_profile(profile_2); - auto sut = handler.validate_tx_profile(profile_1, *evses[evse_id]); + auto profile_1 = create_charging_profile(DEFAULT_PROFILE_ID, ChargingProfilePurposeEnum::TxProfile, + create_charge_schedule(ChargingRateUnitEnum::A), transaction_id, + ChargingProfileKindEnum::Absolute, same_stack_level); + auto profile_2 = create_charging_profile(DEFAULT_PROFILE_ID + 1, ChargingProfilePurposeEnum::TxProfile, + create_charge_schedule(ChargingRateUnitEnum::A), transaction_id, + ChargingProfileKindEnum::Absolute, same_stack_level); + handler.add_profile(DEFAULT_EVSE_ID, profile_2); + auto sut = handler.validate_tx_profile(profile_1, *evses[DEFAULT_EVSE_ID]); EXPECT_THAT(sut, testing::Eq(ProfileValidationResultEnum::TxProfileConflictingStackLevel)); } TEST_F(ChargepointTestFixtureV201, K01FR06_IfTxProfileHasDifferentTransactionButSameStackLevelAsAnotherTxProfile_ThenProfileIsValid) { - create_evse_with_id(evse_id); + create_evse_with_id(DEFAULT_EVSE_ID); std::string transaction_id = uuid(); std::string different_transaction_id = uuid(); - open_evse_transaction(evse_id, transaction_id); + open_evse_transaction(DEFAULT_EVSE_ID, transaction_id); auto same_stack_level = 42; - auto profile_1 = - create_tx_profile(create_charge_schedule(ChargingRateUnitEnum::A), transaction_id, same_stack_level); - auto profile_2 = - create_tx_profile(create_charge_schedule(ChargingRateUnitEnum::A), different_transaction_id, same_stack_level); - handler.add_profile(profile_2); - auto sut = handler.validate_tx_profile(profile_1, *evses[evse_id]); + auto profile_1 = create_charging_profile(DEFAULT_PROFILE_ID, ChargingProfilePurposeEnum::TxProfile, + create_charge_schedule(ChargingRateUnitEnum::A), transaction_id, + ChargingProfileKindEnum::Absolute, same_stack_level); + auto profile_2 = create_charging_profile(DEFAULT_PROFILE_ID + 1, ChargingProfilePurposeEnum::TxProfile, + create_charge_schedule(ChargingRateUnitEnum::A), different_transaction_id, + ChargingProfileKindEnum::Absolute, same_stack_level); + handler.add_profile(DEFAULT_EVSE_ID, profile_2); + auto sut = handler.validate_tx_profile(profile_1, *evses[DEFAULT_EVSE_ID]); EXPECT_THAT(sut, testing::Eq(ProfileValidationResultEnum::Valid)); } TEST_F(ChargepointTestFixtureV201, K01FR06_IfTxProfileHasSameTransactionButDifferentStackLevelAsAnotherTxProfile_ThenProfileIsValid) { - create_evse_with_id(evse_id); + create_evse_with_id(DEFAULT_EVSE_ID); std::string same_transaction_id = uuid(); - open_evse_transaction(evse_id, same_transaction_id); + open_evse_transaction(DEFAULT_EVSE_ID, same_transaction_id); auto stack_level_1 = 42; auto stack_level_2 = 43; - auto profile_1 = - create_tx_profile(create_charge_schedule(ChargingRateUnitEnum::A), same_transaction_id, stack_level_1); - auto profile_2 = - create_tx_profile(create_charge_schedule(ChargingRateUnitEnum::A), same_transaction_id, stack_level_2); - handler.add_profile(profile_2); - auto sut = handler.validate_tx_profile(profile_1, *evses[evse_id]); + + auto profile_1 = create_charging_profile(DEFAULT_PROFILE_ID, ChargingProfilePurposeEnum::TxProfile, + create_charge_schedule(ChargingRateUnitEnum::A), same_transaction_id, + ChargingProfileKindEnum::Absolute, stack_level_1); + auto profile_2 = create_charging_profile(DEFAULT_PROFILE_ID + 1, ChargingProfilePurposeEnum::TxProfile, + create_charge_schedule(ChargingRateUnitEnum::A), same_transaction_id, + ChargingProfileKindEnum::Absolute, stack_level_2); + + handler.add_profile(DEFAULT_EVSE_ID, profile_2); + auto sut = handler.validate_tx_profile(profile_1, *evses[DEFAULT_EVSE_ID]); EXPECT_THAT(sut, testing::Eq(ProfileValidationResultEnum::Valid)); } TEST_F(ChargepointTestFixtureV201, K01FR19_NumberPhasesOtherThan1AndPhaseToUseSet_ThenProfileInvalid) { auto periods = create_charging_schedule_periods_with_phases(0, 0, 1); - auto profile = create_tx_profile( - create_charge_schedule(ChargingRateUnitEnum::A, periods, ocpp::DateTime("2024-01-17T17:00:00")), uuid(), 1, - ChargingProfileKindEnum::Relative); + auto profile = create_charging_profile( + DEFAULT_PROFILE_ID, ChargingProfilePurposeEnum::TxProfile, + create_charge_schedule(ChargingRateUnitEnum::A, periods, ocpp::DateTime("2024-01-17T17:00:00")), uuid(), + ChargingProfileKindEnum::Absolute, 1); auto sut = handler.validate_profile_schedules(profile); @@ -293,7 +314,8 @@ TEST_F(ChargepointTestFixtureV201, K01FR19_NumberPhasesOtherThan1AndPhaseToUseSe } TEST_F(ChargepointTestFixtureV201, K01_IfChargingSchedulePeriodsAreMissing_ThenProfileIsInvalid) { - auto profile = create_tx_profile(create_charge_schedule(ChargingRateUnitEnum::A), uuid()); + auto profile = create_charging_profile(DEFAULT_PROFILE_ID, ChargingProfilePurposeEnum::TxProfile, + create_charge_schedule(ChargingRateUnitEnum::A), uuid()); auto sut = handler.validate_profile_schedules(profile); @@ -302,7 +324,8 @@ TEST_F(ChargepointTestFixtureV201, K01_IfChargingSchedulePeriodsAreMissing_ThenP TEST_F(ChargepointTestFixtureV201, K01FR31_IfStartPeriodOfFirstChargingSchedulePeriodIsNotZero_ThenProfileIsInvalid) { auto periods = create_charging_schedule_periods(1); - auto profile = create_tx_profile(create_charge_schedule(ChargingRateUnitEnum::A, periods), uuid()); + auto profile = create_charging_profile(DEFAULT_PROFILE_ID, ChargingProfilePurposeEnum::TxProfile, + create_charge_schedule(ChargingRateUnitEnum::A, periods), uuid()); auto sut = handler.validate_profile_schedules(profile); @@ -311,7 +334,8 @@ TEST_F(ChargepointTestFixtureV201, K01FR31_IfStartPeriodOfFirstChargingScheduleP TEST_F(ChargepointTestFixtureV201, K01FR35_IfChargingSchedulePeriodsAreNotInChonologicalOrder_ThenProfileIsInvalid) { auto periods = create_charging_schedule_periods({0, 2, 1}); - auto profile = create_tx_profile(create_charge_schedule(ChargingRateUnitEnum::A, periods), uuid()); + auto profile = create_charging_profile(DEFAULT_PROFILE_ID, ChargingProfilePurposeEnum::TxProfile, + create_charge_schedule(ChargingRateUnitEnum::A, periods), uuid()); auto sut = handler.validate_profile_schedules(profile); @@ -321,8 +345,9 @@ TEST_F(ChargepointTestFixtureV201, K01FR35_IfChargingSchedulePeriodsAreNotInChon TEST_F(ChargepointTestFixtureV201, K01FR40_IfChargingProfileKindIsAbsoluteAndStartScheduleDoesNotExist_ThenProfileIsInvalid) { auto periods = create_charging_schedule_periods(0); - auto profile = create_tx_profile(create_charge_schedule(ChargingRateUnitEnum::A, periods), uuid(), 1, - ChargingProfileKindEnum::Absolute); + auto profile = create_charging_profile(DEFAULT_PROFILE_ID, ChargingProfilePurposeEnum::TxProfile, + create_charge_schedule(ChargingRateUnitEnum::A, periods), uuid(), + ChargingProfileKindEnum::Absolute, 1); auto sut = handler.validate_profile_schedules(profile); @@ -332,8 +357,9 @@ TEST_F(ChargepointTestFixtureV201, TEST_F(ChargepointTestFixtureV201, K01FR40_IfChargingProfileKindIsRecurringAndStartScheduleDoesNotExist_ThenProfileIsInvalid) { auto periods = create_charging_schedule_periods(0); - auto profile = create_tx_profile(create_charge_schedule(ChargingRateUnitEnum::A, periods), uuid(), 1, - ChargingProfileKindEnum::Recurring); + auto profile = create_charging_profile(DEFAULT_PROFILE_ID, ChargingProfilePurposeEnum::TxProfile, + create_charge_schedule(ChargingRateUnitEnum::A, periods), uuid(), + ChargingProfileKindEnum::Recurring, 1); auto sut = handler.validate_profile_schedules(profile); @@ -343,13 +369,54 @@ TEST_F(ChargepointTestFixtureV201, TEST_F(ChargepointTestFixtureV201, K01FR41_IfChargingProfileKindIsRelativeAndStartScheduleDoesExist_ThenProfileIsInvalid) { auto periods = create_charging_schedule_periods(0); - auto profile = create_tx_profile( - create_charge_schedule(ChargingRateUnitEnum::A, periods, ocpp::DateTime("2024-01-17T17:00:00")), uuid(), 1, - ChargingProfileKindEnum::Relative); + auto profile = create_charging_profile( + DEFAULT_PROFILE_ID, ChargingProfilePurposeEnum::TxProfile, + create_charge_schedule(ChargingRateUnitEnum::A, periods, ocpp::DateTime("2024-01-17T17:00:00")), uuid(), + ChargingProfileKindEnum::Relative, 1); auto sut = handler.validate_profile_schedules(profile); EXPECT_THAT(sut, testing::Eq(ProfileValidationResultEnum::ChargingProfileExtraneousStartSchedule)); } +/* + * 0 - For K01.FR.52, we return DuplicateTxDefaultProfileFound if an existing profile is on an EVSE, + * and a new profile with the same stack level and different profile ID will be associated with EVSE ID = 0. + * 1 - For K01.FR.53, we return DuplicateTxDefaultProfileFound if an existing profile is on an EVSE ID = 0, + * and a new profile with the same stack level and different profile ID will be associated with an EVSE. + * 2-7 - We return Valid for any other case, such as using the same EVSE or using the same profile ID. + */ + +INSTANTIATE_TEST_SUITE_P( + TxDefaultProfileValidationV201_Param_Test_Instantiate, ChargepointTestFixtureV201, + testing::Values(std::make_tuple(DEFAULT_EVSE_ID, STATION_WIDE_ID, DEFAULT_PROFILE_ID + 1, DEFAULT_STACK_LEVEL, + ProfileValidationResultEnum::DuplicateTxDefaultProfileFound), + std::make_tuple(STATION_WIDE_ID, DEFAULT_EVSE_ID, DEFAULT_PROFILE_ID + 1, DEFAULT_STACK_LEVEL, + ProfileValidationResultEnum::DuplicateTxDefaultProfileFound), + std::make_tuple(STATION_WIDE_ID, STATION_WIDE_ID, DEFAULT_PROFILE_ID + 1, DEFAULT_STACK_LEVEL, + ProfileValidationResultEnum::Valid), + std::make_tuple(DEFAULT_EVSE_ID, DEFAULT_EVSE_ID, DEFAULT_PROFILE_ID + 1, DEFAULT_STACK_LEVEL, + ProfileValidationResultEnum::Valid), + std::make_tuple(DEFAULT_EVSE_ID, STATION_WIDE_ID, DEFAULT_PROFILE_ID, DEFAULT_STACK_LEVEL, + ProfileValidationResultEnum::Valid), + std::make_tuple(STATION_WIDE_ID, DEFAULT_EVSE_ID, DEFAULT_PROFILE_ID, DEFAULT_STACK_LEVEL, + ProfileValidationResultEnum::Valid), + std::make_tuple(DEFAULT_EVSE_ID, STATION_WIDE_ID, DEFAULT_PROFILE_ID + 1, DEFAULT_STACK_LEVEL + 1, + ProfileValidationResultEnum::Valid), + std::make_tuple(STATION_WIDE_ID, DEFAULT_EVSE_ID, DEFAULT_PROFILE_ID + 1, DEFAULT_STACK_LEVEL + 1, + ProfileValidationResultEnum::Valid))); + +TEST_P(ChargepointTestFixtureV201, K01FR52_and_K01FR53_TxDefaultProfileValidationV201Tests) { + auto [existing_evse_id, added_evse_id, added_profile_id, added_stack_level, expected] = GetParam(); + install_profile_on_evse(existing_evse_id, DEFAULT_PROFILE_ID); + + create_evse_with_id(added_evse_id); + auto profile = create_charging_profile(added_profile_id, ChargingProfilePurposeEnum::TxDefaultProfile, + create_charge_schedule(ChargingRateUnitEnum::A), uuid(), + ChargingProfileKindEnum::Absolute, added_stack_level); + auto sut = handler.validate_tx_default_profile(profile, *evses[added_evse_id]); + + EXPECT_THAT(sut, testing::Eq(expected)); +} + } // namespace ocpp::v201