# SPDX-License-Identifier: Apache-2.0 # Copyright Pionix GmbH and Contributors to EVerest from datetime import datetime, timezone import pytest import logging from unittest.mock import ANY from everest.testing.ocpp_utils.central_system import CentralSystem from everest.testing.ocpp_utils.charge_point_v201 import ChargePoint201 from everest.testing.ocpp_utils.charge_point_utils import wait_for_and_validate, TestUtility from everest.testing.ocpp_utils.fixtures import charge_point_v201 from everest.testing.core_utils.controller.test_controller_interface import TestController from everest.testing.core_utils._configuration.libocpp_configuration_helper import GenericOCPP2XConfigAdjustment from everest_test_utils import * from ocpp.v201.enums import (IdTokenEnumType as IdTokenTypeEnum, ReserveNowStatusEnumType, ConnectorStatusEnumType, OperationalStatusEnumType, CancelReservationStatusEnumType, SetVariableStatusEnumType, RequestStartStopStatusEnumType) from ocpp.v201.datatypes import * from ocpp.v201 import call as call_201 from ocpp.v201 import call_result as call_result201 from validations import ( validate_remote_start_stop_transaction, wait_for_callerror_and_validate) from ocpp.routing import on, create_route_map @pytest.mark.asyncio @pytest.mark.ocpp_version("ocpp2.0.1") async def test_reservation_local_start_tx( test_config: OcppTestConfiguration, charge_point_v201: ChargePoint201, test_controller: TestController, test_utility: TestUtility, ): """ Test making a reservation and start a transaction on the reserved evse id with the correct id token. """ logging.info("######### test_reservation_local_start_tx #########") t = datetime.now(timezone.utc) + timedelta(minutes=10) await charge_point_v201.reserve_now_req( id=0, id_token=IdTokenType(id_token=test_config.authorization_info.valid_id_tag_1, type=IdTokenTypeEnum.iso14443), expiry_date_time=t.isoformat(), evse_id=1 ) # expect ReserveNow response with status accepted assert await wait_for_and_validate( test_utility, charge_point_v201, "ReserveNow", call_result201.ReserveNow(ReserveNowStatusEnumType.accepted), ) # expect StatusNotification with status reserved assert await wait_for_and_validate( test_utility, charge_point_v201, "StatusNotification", call_201.StatusNotification( ANY, ConnectorStatusEnumType.reserved, 1, 1 ), ) # swipe invalid id tag test_controller.swipe(test_config.authorization_info.invalid_id_tag) # swipe valid id tag to authorize test_controller.swipe(test_config.authorization_info.valid_id_tag_1) # expect StatusNotification with status available (reservation is now used) assert await wait_for_and_validate( test_utility, charge_point_v201, "StatusNotification", call_201.StatusNotification( ANY, ConnectorStatusEnumType.available, 1, 1 ), ) # start charging session test_controller.plug_in() # expect TransactionEvent with event type Started and the reservation id. assert await wait_for_and_validate( test_utility, charge_point_v201, "TransactionEvent", {"eventType": "Started", "reservationId": 0} ) assert await wait_for_and_validate( test_utility, charge_point_v201, "TransactionEvent", {"eventType": "Updated"} ) # expect StatusNotification with status occupied assert await wait_for_and_validate( test_utility, charge_point_v201, "StatusNotification", call_201.StatusNotification( ANY, 'Occupied', 1, 1 ), ) @pytest.mark.asyncio @pytest.mark.ocpp_version("ocpp2.0.1") async def test_reservation_local_start_tx_plugin_first( test_config: OcppTestConfiguration, charge_point_v201: ChargePoint201, test_controller: TestController, test_utility: TestUtility, ): """ Test making a reservation and start a transaction on the reserved evse id with the correct id token. """ logging.info("######### test_reservation_local_start_tx #########") t = datetime.now(timezone.utc) + timedelta(minutes=10) await charge_point_v201.reserve_now_req( id=0, id_token=IdTokenType(id_token=test_config.authorization_info.valid_id_tag_1, type=IdTokenTypeEnum.iso14443), expiry_date_time=t.isoformat(), evse_id=1 ) # expect ReserveNow response with status accepted assert await wait_for_and_validate( test_utility, charge_point_v201, "ReserveNow", call_result201.ReserveNow(ReserveNowStatusEnumType.accepted), ) # expect StatusNotification with status reserved assert await wait_for_and_validate( test_utility, charge_point_v201, "StatusNotification", call_201.StatusNotification( ANY, ConnectorStatusEnumType.reserved, 1, 1 ), ) test_utility.messages.clear() # start charging session test_controller.plug_in() await asyncio.sleep(2) test_utility.messages.clear() # swipe valid id tag that belongs to this reservation to authorize test_controller.swipe(test_config.authorization_info.valid_id_tag_1) # expect StatusNotification with status available (reservation is now used) assert await wait_for_and_validate( test_utility, charge_point_v201, "StatusNotification", call_201.StatusNotification( ANY, ConnectorStatusEnumType.occupied, 1, 1 ), ) # expect TransactionEvent with event type Started and the reservation id. assert await wait_for_and_validate( test_utility, charge_point_v201, "TransactionEvent", {"eventType": "Started", "reservationId": 0} ) assert await wait_for_and_validate( test_utility, charge_point_v201, "TransactionEvent", {"eventType": "Updated"} ) @pytest.mark.asyncio @pytest.mark.ocpp_version("ocpp2.0.1") async def test_reservation_plug_in_other_idtoken( test_config: OcppTestConfiguration, charge_point_v201: ChargePoint201, test_controller: TestController, test_utility: TestUtility, ): """ Test making a reservation and start a transaction on the reserved evse id with the wrong id token, plug in first. """ logging.info("######### test_reservation_plug_in_other_idtoken #########") t = datetime.now(timezone.utc) + timedelta(minutes=10) await charge_point_v201.reserve_now_req( id=0, id_token=IdTokenType(id_token=test_config.authorization_info.valid_id_tag_1, type=IdTokenTypeEnum.iso14443), expiry_date_time=t.isoformat(), evse_id=1 ) # expect ReserveNow response with status accepted assert await wait_for_and_validate( test_utility, charge_point_v201, "ReserveNow", call_result201.ReserveNow(ReserveNowStatusEnumType.accepted), ) # expect StatusNotification with status reserved assert await wait_for_and_validate( test_utility, charge_point_v201, "StatusNotification", call_201.StatusNotification( ANY, ConnectorStatusEnumType.reserved, 1, 1 ), ) test_utility.messages.clear() # start charging session test_controller.plug_in() # No StatusNotification with status occupied should be sent assert not await wait_for_and_validate( test_utility, charge_point_v201, "StatusNotification", call_201.StatusNotification( ANY, ConnectorStatusEnumType.occupied, 1, 1 ), timeout=5 ) test_utility.messages.clear() # swipe invalid id tag test_controller.swipe(test_config.authorization_info.valid_id_tag_2) assert not await wait_for_and_validate( test_utility, charge_point_v201, "StatusNotification", call_201.StatusNotification( ANY, ConnectorStatusEnumType.occupied, 1, 1 ), timeout=5 ) test_utility.messages.clear() # swipe valid id tag to authorize test_controller.swipe(test_config.authorization_info.valid_id_tag_1) # expect TransactionEvent with event type Started and the reservation id. assert await wait_for_and_validate( test_utility, charge_point_v201, "TransactionEvent", {"eventType": "Started", "reservationId": 0} ) assert await wait_for_and_validate( test_utility, charge_point_v201, "TransactionEvent", {"eventType": "Updated"} ) # expect StatusNotification with status occupied assert await wait_for_and_validate( test_utility, charge_point_v201, "StatusNotification", call_201.StatusNotification( ANY, 'Occupied', 1, 1 ), ) @pytest.mark.asyncio @pytest.mark.ocpp_version("ocpp2.0.1") async def test_reservation_remote_start_tx( test_config: OcppTestConfiguration, charge_point_v201: ChargePoint201, test_controller: TestController, test_utility: TestUtility, ): """ Test making a reservation and start a remote transaction on the reserved evse id with the correct id token. """ logging.info("######### test_reservation_remote_start_tx #########") t = datetime.now(timezone.utc) + timedelta(minutes=10) # Make reservation for evse id 1. await charge_point_v201.reserve_now_req( id=0, id_token=IdTokenType(id_token=test_config.authorization_info.valid_id_tag_1, type=IdTokenTypeEnum.iso14443), expiry_date_time=t.isoformat(), evse_id=1 ) # expect ReserveNow response with status accepted assert await wait_for_and_validate( test_utility, charge_point_v201, "ReserveNow", call_result201.ReserveNow(ReserveNowStatusEnumType.accepted), ) # expect StatusNotification with status reserved assert await wait_for_and_validate( test_utility, charge_point_v201, "StatusNotification", call_201.StatusNotification( ANY, ConnectorStatusEnumType.reserved, 1, 1 ), ) # send start transaction request await charge_point_v201.request_start_transaction_req( id_token=IdTokenType(id_token=test_config.authorization_info.valid_id_tag_1, type=IdTokenTypeEnum.iso14443), remote_start_id=1, evse_id=1 ) # Which should be accepted assert await wait_for_and_validate( test_utility, charge_point_v201, "RequestStartTransaction", call_result201.RequestStartTransaction(status="Accepted"), validate_remote_start_stop_transaction, ) # start charging session test_controller.plug_in() # expect StatusNotification with status available (because reservation is 'used') assert await wait_for_and_validate( test_utility, charge_point_v201, "StatusNotification", call_201.StatusNotification( ANY, ConnectorStatusEnumType.available, 1, 1 ), ) # expect StatusNotification with status occupied assert await wait_for_and_validate( test_utility, charge_point_v201, "StatusNotification", call_201.StatusNotification( ANY, 'Occupied', 1, 1 ), ) # expect StartTransaction with the given reservation id assert await wait_for_and_validate( test_utility, charge_point_v201, "TransactionEvent", {"eventType": "Started", "reservationId": 0} ) assert await wait_for_and_validate( test_utility, charge_point_v201, "TransactionEvent", {"eventType": "Updated"} ) @pytest.mark.asyncio @pytest.mark.ocpp_version("ocpp2.0.1") async def test_reservation_connector_expire( test_config: OcppTestConfiguration, charge_point_v201: ChargePoint201, test_controller: TestController, test_utility: TestUtility, ): """ Test that a reservation can expire. """ logging.info("######### test_reservation_connector_expire #########") # Make a reservation with an expiry time ten seconds from now. t = datetime.now(timezone.utc) + timedelta(seconds=10) await charge_point_v201.reserve_now_req( id=5, id_token=IdTokenType(id_token=test_config.authorization_info.valid_id_tag_1, type=IdTokenTypeEnum.iso14443), expiry_date_time=t.isoformat(), evse_id=1 ) # Reservation is accepted assert await wait_for_and_validate( test_utility, charge_point_v201, "ReserveNow", call_result201.ReserveNow(ReserveNowStatusEnumType.accepted), ) # expect StatusNotification with status reserved assert await wait_for_and_validate( test_utility, charge_point_v201, "StatusNotification", call_201.StatusNotification( ANY, ConnectorStatusEnumType.reserved, 1, 1 ), ) # Request to start transaction for the reserved evse but with another id token. await charge_point_v201.request_start_transaction_req( id_token=IdTokenType(id_token=test_config.authorization_info.valid_id_tag_2, type=IdTokenTypeEnum.iso14443), remote_start_id=1, evse_id=1 ) # This will not succeed. assert await wait_for_and_validate( test_utility, charge_point_v201, "RequestStartTransaction", call_result201.RequestStartTransaction(status="Rejected"), validate_remote_start_stop_transaction, ) # So we wait until the reservation is expired. assert await wait_for_and_validate( test_utility, charge_point_v201, "ReservationStatusUpdate", call_201.ReservationStatusUpdate( 5, "Expired" ), ) # expect StatusNotification with status available assert await wait_for_and_validate( test_utility, charge_point_v201, "StatusNotification", call_201.StatusNotification( ANY, ConnectorStatusEnumType.available, 1, 1 ), ) # Try to start a transaction now on the previously reserved evse (which reservation is now expired). await charge_point_v201.request_start_transaction_req( id_token=IdTokenType(id_token=test_config.authorization_info.valid_id_tag_2, type=IdTokenTypeEnum.iso14443), remote_start_id=1, evse_id=1 ) # Now it succeeds. assert await wait_for_and_validate( test_utility, charge_point_v201, "RequestStartTransaction", call_result201.RequestStartTransaction(status="Accepted"), validate_remote_start_stop_transaction, ) # Start charging session test_controller.plug_in() # expect StatusNotification with status occupied assert await wait_for_and_validate( test_utility, charge_point_v201, "StatusNotification", call_201.StatusNotification( ANY, 'Occupied', 1, 1 ), ) # expect TransactionEvent with event type 'Started' assert await wait_for_and_validate( test_utility, charge_point_v201, "TransactionEvent", {"eventType": "Started"} ) assert await wait_for_and_validate( test_utility, charge_point_v201, "TransactionEvent", {"eventType": "Updated"} ) @pytest.mark.asyncio @pytest.mark.ocpp_version("ocpp2.0.1") async def test_reservation_connector_faulted( test_config: OcppTestConfiguration, charge_point_v201: ChargePoint201, test_controller: TestController, test_utility: TestUtility, ): """ Test if an evse can be reserved when the evse status is 'Faulted' """ logging.info("######### test_reservation_connector_faulted #########") # Set evse in state 'faulted' test_controller.raise_error("MREC6UnderVoltage", 1) assert await wait_for_and_validate( test_utility, charge_point_v201, "StatusNotification", call_201.StatusNotification( ANY, 'Faulted', 1, 1 ), ) await asyncio.sleep(1) t = datetime.now(timezone.utc) + timedelta(minutes=10) # Send a reserve new request await charge_point_v201.reserve_now_req( id=0, id_token=IdTokenType(id_token=test_config.authorization_info.valid_id_tag_1, type=IdTokenTypeEnum.iso14443), expiry_date_time=t.isoformat(), evse_id=1 ) # which should return 'faulted' assert await wait_for_and_validate( test_utility, charge_point_v201, "ReserveNow", call_result201.ReserveNow(ReserveNowStatusEnumType.faulted), ) test_controller.clear_error() test_utility.messages.clear() assert await wait_for_and_validate( test_utility, charge_point_v201, "StatusNotification", call_201.StatusNotification( ANY, 'Available', 1, 1 ), ) @pytest.mark.asyncio @pytest.mark.ocpp_version("ocpp2.0.1") async def test_reservation_connector_faulted_after_reservation( test_config: OcppTestConfiguration, charge_point_v201: ChargePoint201, test_controller: TestController, test_utility: TestUtility, ): """ Test if a reservation is cancelled after the evse status is 'Faulted' """ logging.info( "######### test_reservation_connector_faulted_after_reservation #########") t = datetime.now(timezone.utc) + timedelta(minutes=10) await charge_point_v201.reserve_now_req( id=42, id_token=IdTokenType(id_token=test_config.authorization_info.valid_id_tag_1, type=IdTokenTypeEnum.iso14443), expiry_date_time=t.isoformat(), evse_id=1 ) # Set evse in state 'faulted' test_controller.raise_error("MREC6UnderVoltage", 1) # This should result in the reservation being cancelled. assert await wait_for_and_validate( test_utility, charge_point_v201, "ReservationStatusUpdate", call_201.ReservationStatusUpdate( 42, "Removed" ), ) test_controller.clear_error() test_utility.messages.clear() assert await wait_for_and_validate( test_utility, charge_point_v201, "StatusNotification", call_201.StatusNotification( ANY, 'Available', 1, 1 ), ) @pytest.mark.asyncio @pytest.mark.ocpp_version("ocpp2.0.1") async def test_reservation_connector_occupied( test_config: OcppTestConfiguration, charge_point_v201: ChargePoint201, test_controller: TestController, test_utility: TestUtility, ): """ Try to make a reservation while a evse is occupied. """ logging.info("######### test_reservation_connector_occupied #########") # start charging session test_controller.plug_in() # expect StatusNotification with status occupied assert await wait_for_and_validate( test_utility, charge_point_v201, "StatusNotification", call_201.StatusNotification( ANY, 'Occupied', 1, 1 ), ) t = datetime.now(timezone.utc) + timedelta(minutes=10) await asyncio.sleep(2) # Request reservation await charge_point_v201.reserve_now_req( id=0, id_token=IdTokenType(id_token=test_config.authorization_info.valid_id_tag_1, type=IdTokenTypeEnum.iso14443), expiry_date_time=t.isoformat(), evse_id=1 ) # expect ReserveNow response with status occupied assert await wait_for_and_validate( test_utility, charge_point_v201, "ReserveNow", call_result201.ReserveNow(ReserveNowStatusEnumType.occupied), ) @pytest.mark.asyncio @pytest.mark.ocpp_version("ocpp2.0.1") async def test_reservation_connector_unavailable( test_config: OcppTestConfiguration, charge_point_v201: ChargePoint201, test_utility: TestUtility, ): """ Test making a reservation with an unavailable connector, this should return 'unavailable' on a reserve now request. """ logging.info("######### test_reservation_connector_unavailable #########") # Set evse id 1 to inoperative. await charge_point_v201.change_availablility_req( evse={'id': 1}, operational_status=OperationalStatusEnumType.inoperative ) t = datetime.now(timezone.utc) + timedelta(seconds=30) # Make a reservation for evse id 1. await charge_point_v201.reserve_now_req( id=0, id_token=IdTokenType(id_token=test_config.authorization_info.valid_id_tag_1, type=IdTokenTypeEnum.iso14443), expiry_date_time=t.isoformat(), evse_id=1 ) # Which should fail (ReserveNow response 'Unavailable'). assert await wait_for_and_validate( test_utility, charge_point_v201, "ReserveNow", call_result201.ReserveNow(ReserveNowStatusEnumType.unavailable), ) @pytest.mark.asyncio @pytest.mark.ocpp_version("ocpp2.0.1") @pytest.mark.ocpp_config_adaptions( GenericOCPP2XConfigAdjustment( [ ( OCPP2XConfigVariableIdentifier( "ReservationCtrlr", "ReservationCtrlrAvailable", "Actual" ), "false", ) ] ) ) async def test_reservation_connector_rejected( test_config: OcppTestConfiguration, charge_point_v201: ChargePoint201, test_utility: TestUtility, test_controller: TestController, ): """ Test if reservation is rejected with the reservation ctrlr is not available. """ logging.info("######### test_reservation_connector_rejected #########") t = datetime.now(timezone.utc) + timedelta(seconds=10) # Try to make reservation. await charge_point_v201.reserve_now_req( id=5, id_token=IdTokenType(id_token=test_config.authorization_info.valid_id_tag_1, type=IdTokenTypeEnum.iso14443), expiry_date_time=t.isoformat(), evse_id=1 ) assert await wait_for_callerror_and_validate(test_utility, charge_point_v201, "NotImplemented") @pytest.mark.asyncio @pytest.mark.ocpp_version("ocpp2.0.1") @pytest.mark.ocpp_config_adaptions( GenericOCPP2XConfigAdjustment( [ ( OCPP2XConfigVariableIdentifier( "ReservationCtrlr", "ReservationCtrlrNonEvseSpecific", "Actual" ), "false", ) ] ) ) async def test_reservation_non_evse_specific_rejected( test_config: OcppTestConfiguration, charge_point_v201: ChargePoint201, test_utility: TestUtility, test_controller: TestController, ): """ Try to make a non-evse specific reservation, while that is not allowed according to the settings. That should fail. """ logging.info( "######### test_reservation_non_evse_specific_rejected #########") t = datetime.now(timezone.utc) + timedelta(seconds=30) # Try to make a reservation without evse id. await charge_point_v201.reserve_now_req( id=5, id_token=IdTokenType(id_token=test_config.authorization_info.valid_id_tag_1, type=IdTokenTypeEnum.iso14443), expiry_date_time=t.isoformat() ) # expect ReserveNow respone with status rejected assert await wait_for_and_validate( test_utility, charge_point_v201, "ReserveNow", call_result201.ReserveNow(ReserveNowStatusEnumType.rejected), ) @pytest.mark.asyncio @pytest.mark.ocpp_version("ocpp2.0.1") @pytest.mark.ocpp_config_adaptions( GenericOCPP2XConfigAdjustment( [ ( OCPP2XConfigVariableIdentifier( "ReservationCtrlr", "ReservationCtrlrNonEvseSpecific", "Actual" ), "true", ) ] ) ) async def test_reservation_non_evse_specific_accepted( test_config: OcppTestConfiguration, charge_point_v201: ChargePoint201, test_utility: TestUtility, test_controller: TestController, ): """ Try to make a non evse specific reservation. This should succeed, according to the settings (devicemodel). """ logging.info( "######### test_reservation_non_evse_specific_accepted #########") t = datetime.now(timezone.utc) + timedelta(seconds=30) # Make reservation without evse id. await charge_point_v201.reserve_now_req( id=5, id_token=IdTokenType(id_token=test_config.authorization_info.valid_id_tag_1, type=IdTokenTypeEnum.iso14443), expiry_date_time=t.isoformat() ) # This should be accepted. assert await wait_for_and_validate( test_utility, charge_point_v201, "ReserveNow", call_result201.ReserveNow(ReserveNowStatusEnumType.accepted), ) # swipe valid id tag to authorize test_controller.swipe(test_config.authorization_info.valid_id_tag_1) # start charging session test_controller.plug_in() # expect TransactionEvent with event type 'Started' and the correct reservation id. assert await wait_for_and_validate( test_utility, charge_point_v201, "TransactionEvent", {"eventType": "Started", "reservationId": 5} ) assert await wait_for_and_validate( test_utility, charge_point_v201, "TransactionEvent", {"eventType": "Updated"} ) # expect StatusNotification with status occupied assert await wait_for_and_validate( test_utility, charge_point_v201, "StatusNotification", call_201.StatusNotification( ANY, 'Occupied', 1, 1 ), ) @pytest.mark.asyncio @pytest.mark.ocpp_version("ocpp2.0.1") @pytest.mark.ocpp_config_adaptions( GenericOCPP2XConfigAdjustment( [ ( OCPP2XConfigVariableIdentifier( "ReservationCtrlr", "ReservationCtrlrNonEvseSpecific", "Actual" ), "true", ) ] ) ) async def test_reservation_non_evse_specific_accepted_multiple( test_config: OcppTestConfiguration, charge_point_v201: ChargePoint201, test_utility: TestUtility, test_controller: TestController, ): """ Try to make a non evse specific reservation. This should succeed, according to the settings (devicemodel). When making multiple reservations, as soon as there are as many reservations as evse's available, the evse's should go to occupied. """ logging.info( "######### test_reservation_non_evse_specific_accepted_multiple #########") t = datetime.now(timezone.utc) + timedelta(seconds=30) # Make reservation await charge_point_v201.reserve_now_req( id=5, id_token=IdTokenType(id_token=test_config.authorization_info.valid_id_tag_1, type=IdTokenTypeEnum.iso14443), expiry_date_time=t.isoformat() ) # Expect it to be accepted. assert await wait_for_and_validate( test_utility, charge_point_v201, "ReserveNow", call_result201.ReserveNow(ReserveNowStatusEnumType.accepted), ) # Make another reservation with another reservation id. await charge_point_v201.reserve_now_req( id=6, id_token=IdTokenType(id_token=test_config.authorization_info.valid_id_tag_1, type=IdTokenTypeEnum.iso14443), expiry_date_time=t.isoformat() ) # expect This should be accepted as well. assert await wait_for_and_validate( test_utility, charge_point_v201, "ReserveNow", call_result201.ReserveNow(ReserveNowStatusEnumType.accepted), ) # There are now as many reservations as evse's, so all evse's go to 'reserved'. assert await wait_for_and_validate( test_utility, charge_point_v201, "StatusNotification", call_201.StatusNotification( ANY, ConnectorStatusEnumType.reserved, 1, 1 ), ) assert await wait_for_and_validate( test_utility, charge_point_v201, "StatusNotification", call_201.StatusNotification( ANY, ConnectorStatusEnumType.reserved, 2, 1 ), ) # There are now as many reservations as evse's, so another reservation is not possible. await charge_point_v201.reserve_now_req( id=7, id_token=IdTokenType(id_token=test_config.authorization_info.valid_id_tag_2, type=IdTokenTypeEnum.iso14443), expiry_date_time=t.isoformat() ) # expect ReserveNow response with status occupied assert await wait_for_and_validate( test_utility, charge_point_v201, "ReserveNow", call_result201.ReserveNow(ReserveNowStatusEnumType.occupied), ) # swipe valid id tag to authorize test_controller.swipe(test_config.authorization_info.valid_id_tag_1) # start charging session test_controller.plug_in() # expect TransactionEvent 'Started' with the correct reservation id. assert await wait_for_and_validate( test_utility, charge_point_v201, "TransactionEvent", {"eventType": "Started", "reservationId": 5} ) assert await wait_for_and_validate( test_utility, charge_point_v201, "TransactionEvent", {"eventType": "Updated"} ) # expect StatusNotification with status charging assert await wait_for_and_validate( test_utility, charge_point_v201, "StatusNotification", call_201.StatusNotification( ANY, 'Occupied', 1, 1 ), ) # swipe id tag to de-authorize test_controller.swipe(test_config.authorization_info.valid_id_tag_1) # stop charging session test_controller.plug_out() # expect TransactionEvent 'Ended' with the correct reservation id. assert await wait_for_and_validate( test_utility, charge_point_v201, "TransactionEvent", {"eventType": "Ended"} ) # expect StatusNotification with status 'available' for both evse's as there are now less reservations than # available evse's. assert await wait_for_and_validate( test_utility, charge_point_v201, "StatusNotification", call_201.StatusNotification( ANY, 'Available', 1, 1 ), ) assert await wait_for_and_validate( test_utility, charge_point_v201, "StatusNotification", call_201.StatusNotification( ANY, 'Available', 2, 1 ), ) @pytest.mark.asyncio @pytest.mark.ocpp_version("ocpp2.0.1") async def test_reservation_faulted_state( test_config: OcppTestConfiguration, charge_point_v201: ChargePoint201, test_utility: TestUtility, test_controller: TestController, ): """ Test if making a reservation is possible if the evse is in faulted state. """ logging.info("######### test_reservation_faulted_state #########") test_controller.raise_error("MREC6UnderVoltage", 1) assert await wait_for_and_validate( test_utility, charge_point_v201, "StatusNotification", call_201.StatusNotification( ANY, 'Faulted', 1, 1 ), ) await asyncio.sleep(1) t = datetime.now(timezone.utc) + timedelta(seconds=30) # Try to make the reservation. await charge_point_v201.reserve_now_req( id=0, id_token=IdTokenType(id_token=test_config.authorization_info.valid_id_tag_1, type=IdTokenTypeEnum.iso14443), expiry_date_time=t.isoformat(), evse_id=1 ) # expect ReserveNow response with status 'Faulted' assert await wait_for_and_validate( test_utility, charge_point_v201, "ReserveNow", call_result201.ReserveNow(ReserveNowStatusEnumType.faulted), ) test_controller.clear_error("MREC6UnderVoltage", 1) test_utility.messages.clear() assert await wait_for_and_validate( test_utility, charge_point_v201, "StatusNotification", call_201.StatusNotification( ANY, 'Available', 1, 1 ), ) @pytest.mark.asyncio @pytest.mark.ocpp_version("ocpp2.0.1") async def test_reservation_cancel( test_config: OcppTestConfiguration, charge_point_v201: ChargePoint201, test_controller: TestController, test_utility: TestUtility, ): """ Test if a reservation can be cancelled. """ logging.info("######### test_reservation_cancel #########") t = datetime.now(timezone.utc) + timedelta(minutes=10) # Make the reservation await charge_point_v201.reserve_now_req( id=5, id_token=IdTokenType(id_token=test_config.authorization_info.valid_id_tag_1, type=IdTokenTypeEnum.iso14443), expiry_date_time=t.isoformat(), evse_id=1 ) # Expect it to be accepted. assert await wait_for_and_validate( test_utility, charge_point_v201, "ReserveNow", call_result201.ReserveNow(ReserveNowStatusEnumType.accepted), ) # expect StatusNotification request with status reserved assert await wait_for_and_validate( test_utility, charge_point_v201, "StatusNotification", call_201.StatusNotification( ANY, ConnectorStatusEnumType.reserved, 1, 1 ), ) # Cancel the reservation. await charge_point_v201.cancel_reservation_req(reservation_id=5) # expect CancelReservation response with status accepted assert await wait_for_and_validate( test_utility, charge_point_v201, "CancelReservation", call_result201.CancelReservation( status=CancelReservationStatusEnumType.accepted), ) # expect StatusNotification with status available assert await wait_for_and_validate( test_utility, charge_point_v201, "StatusNotification", call_201.StatusNotification( ANY, ConnectorStatusEnumType.available, 1, 1 ), ) # start charging session test_controller.plug_in() assert await wait_for_and_validate( test_utility, charge_point_v201, "StatusNotification", call_201.StatusNotification( ANY, 'Occupied', 1, 1 ), ) # send request start transaction await charge_point_v201.request_start_transaction_req( id_token=IdTokenType(id_token=test_config.authorization_info.valid_id_tag_1, type=IdTokenTypeEnum.iso14443), remote_start_id=1, evse_id=1 ) # Which should be accepted assert await wait_for_and_validate( test_utility, charge_point_v201, "RequestStartTransaction", call_result201.RequestStartTransaction(status="Accepted"), validate_remote_start_stop_transaction, ) # expect TransactionEvent with eventType 'Started' assert await wait_for_and_validate( test_utility, charge_point_v201, "TransactionEvent", {"eventType": "Started"} ) # expect TransactionEvent with status 'Updated' assert await wait_for_and_validate( test_utility, charge_point_v201, "TransactionEvent", {"eventType": "Updated"} ) @pytest.mark.asyncio @pytest.mark.ocpp_version("ocpp2.0.1") async def test_reservation_cancel_rejected( test_config: OcppTestConfiguration, charge_point_v201: ChargePoint201, test_utility: TestUtility, ): """ Try to cancel a non existing reservation """ logging.info("######### test_reservation_cancel_rejected #########") t = datetime.now(timezone.utc) + timedelta(minutes=10) # Make a reservation with reservation id 5 await charge_point_v201.reserve_now_req( id=5, id_token=IdTokenType(id_token=test_config.authorization_info.valid_id_tag_1, type=IdTokenTypeEnum.iso14443), expiry_date_time=t.isoformat(), evse_id=1 ) # expect ReserveNow response with status accepted assert await wait_for_and_validate( test_utility, charge_point_v201, "ReserveNow", call_result201.ReserveNow(ReserveNowStatusEnumType.accepted), ) # expect StatusNotification with status reserved assert await wait_for_and_validate( test_utility, charge_point_v201, "StatusNotification", call_201.StatusNotification( ANY, ConnectorStatusEnumType.reserved, 1, 1 ), ) # Try to cancel reservation with reservation id 2, which does not exist. await charge_point_v201.cancel_reservation_req(reservation_id=2) # expect CancelReservation response with status rejected assert await wait_for_and_validate( test_utility, charge_point_v201, "CancelReservation", call_result201.CancelReservation( status=CancelReservationStatusEnumType.rejected), ) @pytest.mark.asyncio @pytest.mark.ocpp_version("ocpp2.0.1") async def test_reservation_with_parentid( test_config: OcppTestConfiguration, charge_point_v201: ChargePoint201, test_utility: TestUtility, test_controller: TestController, central_system_v201: CentralSystem ): """ Test reservation with parent id. """ logging.info("######### test_reservation_with_parentid #########") # authorize.conf with parent id tag @on(Action.authorize) def on_authorize(**kwargs): id_tag_info = IdTokenInfoType( status=AuthorizationStatusEnumType.accepted, group_id_token=IdTokenType( id_token=test_config.authorization_info.parent_id_tag, type=IdTokenTypeEnum.iso14443 ), ) return call_result201.Authorize(id_token_info=id_tag_info) setattr(charge_point_v201, "on_authorize", on_authorize) central_system_v201.chargepoint.route_map = create_route_map( central_system_v201.chargepoint ) charge_point_v201.route_map = create_route_map(charge_point_v201) r: call_result201.SetVariables = ( await charge_point_v201.set_config_variables_req( "AuthCtrlr", "AuthorizeRemoteStart", "true" ) ) set_variable_result: SetVariableResultType = SetVariableResultType( **r.set_variable_result[0] ) assert set_variable_result.attribute_status == SetVariableStatusEnumType.accepted # Disable remote authorization so an 'Authorize' request is sent when starting remotely. r: call_result201.SetVariables = ( await charge_point_v201.set_config_variables_req( "AuthCtrlr", "DisableRemoteAuthorization", "false" ) ) set_variable_result: SetVariableResultType = SetVariableResultType( **r.set_variable_result[0] ) assert set_variable_result.attribute_status == SetVariableStatusEnumType.accepted t = datetime.now(timezone.utc) + timedelta(minutes=10) # Make a new reservation. await charge_point_v201.reserve_now_req( evse_id=1, expiry_date_time=t.isoformat(), id_token=IdTokenType(id_token=test_config.authorization_info.valid_id_tag_1, type=IdTokenTypeEnum.iso14443), group_id_token=IdTokenType(id_token=test_config.authorization_info.parent_id_tag, type=IdTokenTypeEnum.iso14443), id=0, ) # expect ReserveNow response with status accepted assert await wait_for_and_validate( test_utility, charge_point_v201, "ReserveNow", call_result201.ReserveNow(ReserveNowStatusEnumType.accepted), ) # expect StatusNotification request with status reserved assert await wait_for_and_validate( test_utility, charge_point_v201, "StatusNotification", call_201.StatusNotification( ANY, ConnectorStatusEnumType.reserved, 1, 1 ), ) # start charging session test_controller.plug_in() # send request start transaction for another id tag than the one from the reservation. await charge_point_v201.request_start_transaction_req( id_token=IdTokenType(id_token=test_config.authorization_info.valid_id_tag_2, type=IdTokenTypeEnum.iso14443), remote_start_id=1, evse_id=1 ) # This is accepted because the reservation has a group id token. assert await wait_for_and_validate( test_utility, charge_point_v201, "RequestStartTransaction", call_result201.RequestStartTransaction(status="Accepted"), validate_remote_start_stop_transaction, ) # expect Authorize request. assert await wait_for_and_validate( test_utility, charge_point_v201, "Authorize", call_201.Authorize(IdTokenType(id_token=test_config.authorization_info.valid_id_tag_2, type=IdTokenTypeEnum.iso14443)), ) # Authorize was accepted because of the correct group id token, transaction is started. r: call_201.TransactionEvent = call_201.TransactionEvent( **await wait_for_and_validate( test_utility, charge_point_v201, "TransactionEvent", {"eventType": "Started"} ) ) transaction = TransactionType(**r.transaction_info) # expect TransactionEvent with eventType 'Updated' assert await wait_for_and_validate( test_utility, charge_point_v201, "TransactionEvent", {"eventType": "Updated"} ) # send request stop transaction await charge_point_v201.request_stop_transaction_req( transaction_id=transaction.transaction_id ) # Which should be accepted. assert await wait_for_and_validate( test_utility, charge_point_v201, "RequestStopTransaction", call_result201.RequestStartTransaction( status=RequestStartStopStatusEnumType.accepted ), ) # And the session should end. assert await wait_for_and_validate( test_utility, charge_point_v201, "TransactionEvent", {"eventType": "Ended"} )