diff --git a/CHANGELOG.md b/CHANGELOG.md index 94e53d9995..97a4024fa8 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -94,6 +94,8 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 - Rewrote CylindricalThermalStorageTest Test from groovy to scala [#646](https://github.com/ie3-institute/simona/issues/646) - Replace mutable var in ChpModelSpec [#1002](https://github.com/ie3-institute/simona/issues/1002) - Move compression of output files into `ResultEventListener`[#965](https://github.com/ie3-institute/simona/issues/965) +- Rewrote StorageModelTest from groovy to scala [#646](https://github.com/ie3-institute/simona/issues/646) +- Updated `ExtEvSimulationClasses` [#898](https://github.com/ie3-institute/simona/issues/898) ### Fixed - Removed a repeated line in the documentation of vn_simona config [#658](https://github.com/ie3-institute/simona/issues/658) diff --git a/docs/readthedocs/requirements.txt b/docs/readthedocs/requirements.txt index 5ed7a8bf17..fcef71943b 100644 --- a/docs/readthedocs/requirements.txt +++ b/docs/readthedocs/requirements.txt @@ -1,5 +1,5 @@ Sphinx==8.1.3 -sphinx-rtd-theme==3.0.1 +sphinx-rtd-theme==3.0.2 sphinxcontrib-plantuml==0.30 myst-parser==4.0.0 markdown-it-py==3.0.0 diff --git a/docs/uml/main/ExtEvSimulationClasses.puml b/docs/uml/main/ExtEvSimulationClasses.puml index 373387d8b8..500f1a8ea2 100644 --- a/docs/uml/main/ExtEvSimulationClasses.puml +++ b/docs/uml/main/ExtEvSimulationClasses.puml @@ -1,177 +1,149 @@ @startuml 'https://plantuml.com/class-diagram skinparam linetype ortho +skinparam nodesep 50 +skinparam ranksep 40 -package ev-simulation { - class ExtLink - class ExternalSampleSim - class EvModelImpl - - ' adding hidden elements for layout purposes - class placeholderA { - w-w-w-w-w-w-w-w-w-w-w-w-w-w-w-w-w-w-w-w-w-w-w-w-w-w-w - } - hide placeholderA - - class placeholderB { - w-w-w-w-w-w-w-w-w-w-w-w-w-w-w-w-w-w-w-w-w-w-w-w-w-w-w-w-w-w-w-w-w-w-w-w-w-w-w-w - } - hide placeholderB +skinparam node { + borderColor Transparent + fontColor Transparent +} - ExtLink -[hidden]> placeholderA - placeholderA -[hidden]> ExternalSampleSim - ExternalSampleSim -[hidden]> placeholderB - placeholderB -[hidden]> EvModelImpl +package MobilitySimulator { + class ExtLink + class MobilitySim + class ElectricVehicle } package simona-api { - ' MIDDLE PART - class ExtEvData { - ~ LinkedBlockingQueue receiveTriggerQueue - - ActorRef dataService - - ActorRef extSimAdapter - + List requestAvailablePublicEvCs() - + List sendEvPositions(EvMovementsMessage evMovementsMessage) - - void sendData(ExtEvMessage msg) - + void queueExtMsg(ExtEvResponseMessage extEvResponse) - } - - class ExtSimAdapterData { - ~ LinkedBlockingQueue receiveTriggerQueue - - ActorRef extSimAdapter - + void queueExtMsg(ExtTrigger trigger) - + void send(ExtTriggerResponse msg) - } - - interface ExtData abstract class ExtSimulation { + - ExtSimAdapterData data + void setup(ExtSimAdapterData data, List adapters) + void run() - # {abstract} List doActivity(long tick) + # {abstract} Optional doActivity(long tick) + # {abstract} Long initialize() } + ExtSimulation --r> ExtData + MobilitySim -|> ExtSimulation + + interface ExtLinkInterface { + + ExtSimulation getExtSimulation() + + List getExtDataSimulations() + } + + ExtLink --|> ExtLinkInterface interface ExtEvSimulation { - + void setExtEvData(ExtEvData evData) + + void setExtEvData(ExtEvData evData) } + MobilitySim --|> ExtEvSimulation + interface ExtDataSimulation - ExternalSampleSim --|> ExtSimulation - ExternalSampleSim --|> ExtEvSimulation + ExtEvSimulation --u|> ExtDataSimulation - ExtSimulation -[hidden]> ExtDataSimulation - ExtEvSimulation -|> ExtDataSimulation - ExtEvSimulation --> ExtEvData + interface EvModel { + + UUID getUuid() + + String getId() + + ComparableQuantity getSRatedAC() + + ComparableQuantity getSRatedDC() + + ComparableQuantity getEStorage() + + ComparableQuantity getStoredEnergy() + + Long getDepartureTick() + + EvModel copyWith(ComparableQuantity newStoredEnergy) + } - ExtSimulation --> ExtData - ExtSimulation --> ExtSimAdapterData + ElectricVehicle --|> EvModel - ExtSimAdapterData -[hidden]> ExtEvData - ExtData <|-- ExtEvData + interface ExtData - ' placeholder - class plchldr1 { - w-w-w-w-w - } - hide plchldr1 - class placeholder2 { - w-w-w-w-w-w-w-w-w-w-w-w-w-w-w - } - hide placeholder2 - class placeholder3 { - w-w-w-w-w-w-w-w-w-w-w - } - hide placeholder3 - class placeholder4 { - w-w-w-w-w-w-w-w-w-w-w-w-w-w-w-w-w-w-w-w-w + class ExtEvData { + + LinkedBlockingQueue receiveTriggerQueue + - ActorRef dataService + - ActorRef extSimAdapter + + Map requestAvailablePublicEvcs() + + Map requestCurrentPrices() + + List requestDepartingEvs(Map> departures) + + void provideArrivingEvs(Map> arrivals, Optional maybeNextTick) + + void sendExtMsg(EvDataMessageFromExt msg) + + void queueExtResponseMsg(EvDataResponseMessageToExt extEvResponse) } - hide placeholder4 - ExtData -[hidden]> plchldr1 - plchldr1 -[hidden]> placeholder4 - placeholder4 -[hidden]> EvMovement - ExtSimulation -[hidden]-> placeholder2 - placeholder2 -[hidden]-> placeholder3 - placeholder3 -[hidden]-> ExtSimAdapterData - placeholder2 -[hidden]> ExtData + node sub2 { + interface EvDataMessageFromExt + interface DataMessageFromExt - ' RIGHT PART - abstract class EvMovement - class Arrival - class Departure - Arrival --|> EvMovement - Departure --|> EvMovement - Arrival -[hidden]> Departure + class ProvideArrivingEvs - interface EvModel { - + UUID getUuid() - + String getId() - + ComparableQuantity getSRated() - + ComparableQuantity getEStorage() - + ComparableQuantity getStoredEnergy() - + EvModel copyWith(ComparableQuantity newStoredEnergy) + class RequestCurrentPrices + + class RequestDepartingEvs + + class RequestEvcsFreeLots } - ExtDataSimulation -[hidden]> Arrival - EvMovement *-> EvModel - EvModelImpl --|> EvModel + RequestDepartingEvs --u|> EvDataMessageFromExt + RequestEvcsFreeLots --u|> EvDataMessageFromExt + ProvideArrivingEvs --u|> EvDataMessageFromExt + RequestCurrentPrices --u|> EvDataMessageFromExt + DataMessageFromExt --d|> EvDataMessageFromExt + ExtEvData -----> EvDataMessageFromExt + ExtEvData --u|> ExtData + ExtEvSimulation -u> ExtEvData - interface EvDataMessageFromExt - class EvMovementsMessage { - - Map> movements - } - class RequestEvcsFreeLots - RequestEvcsFreeLots --|> EvDataMessageFromExt - EvMovementsMessage --|> EvDataMessageFromExt - RequestEvcsFreeLots -[hidden]> EvMovementsMessage - - interface EvDataResponseMessageToExt - class AllDepartedEvsRepsonse { - - Map> movements - } - class ProvideEvcsFreeLots - ProvideEvcsFreeLots --|> EvDataResponseMessageToExt - AllDepartedEvsRepsonse --|> EvDataResponseMessageToExt - ExtEvData -> EvDataMessageFromExt - ExtEvData -> EvDataResponseMessageToExt - EvMovement -[hidden]-> RequestEvcsFreeLots - EvDataMessageFromExt -[hidden]> EvDataResponseMessageToExt - EvMovementsMessage -[hidden]> ProvideEvcsFreeLots - ProvideEvcsFreeLots -[hidden]> AllDepartedEvsRepsonse - - class ScheduleDataServiceMessage { - - ExtEvDataService dataService - } - EvDataResponseMessageToExt -[hidden]> ScheduleDataServiceMessage - ExtEvData -> ScheduleDataServiceMessage + node sub1 { + interface DataResponseMessageToExt + interface EvDataResponseMessageToExt - ' LEFT PART - interface ExtLinkInterface { - + ExtSimulation getExtSimulation() - + List getExtDataSimulations() - } - ExtLinkInterface -[hidden]> ExtSimulation - ExtLink --|> ExtLinkInterface + class ProvideCurrentPrices { + + Map prices + } + + class ProvideDepartingEvs { + + List departedEvs + } + + class ProvideEvcsFreeLots { + + Map evcs + } - interface ExtTrigger - class ActivationMessage { - - Long tick + ProvideEvcsFreeLots --|> EvDataResponseMessageToExt + ProvideDepartingEvs --|> EvDataResponseMessageToExt + ProvideCurrentPrices --|> EvDataResponseMessageToExt + ExtEvData --> EvDataResponseMessageToExt + EvDataResponseMessageToExt --|> DataResponseMessageToExt } - ActivationMessage --|> ExtTrigger - interface ExtTriggerResponse - class CompletionMessage { - - Optional nextActivation + node sub3 { + class ExtSimAdapterData { + + LinkedBlockingQueue receiveTriggerQueue + - ActorRef extSimAdapter + - String[] mainArgs + + void queueExtMsg(ControlMessageToExt msg) + + void send(ControlResponseMessageFromExt msg) + + String[] getMainArgs() + } + + interface ControlMessageToExt + + interface ControlResponseMessageFromExt + + class CompletionMessage + + class ActivationMessage + + ExtSimAdapterData --u> ControlMessageToExt + ExtSimAdapterData --u> ControlResponseMessageFromExt + CompletionMessage --d|> ControlResponseMessageFromExt + ActivationMessage --d|> ControlMessageToExt + ExtSimulation -----> ExtSimAdapterData } - CompletionMessage --|> ExtTriggerResponse - ExtTrigger -[hidden]> ExtTriggerResponse - ExtTrigger <- ExtSimAdapterData - ExtTriggerResponse <- ExtSimAdapterData } package simona { @@ -182,10 +154,22 @@ package simona { class EvcsAgent interface SecondaryData - class EvMovementData { - movements: Set[EvMovement] + + class ArrivingEvs { + + arrivals: Seq[EvModelWrapper] } + class DepartingEvsRequest { + + evcs: UUID + + departingEvs: Seq[EvModelWrapper] + } + + class EvFreeLotsRequest { + + freeLots: Int + } + + abstract EvData + class SimonaSim class Scheduler @@ -202,22 +186,32 @@ package simona { SimonaSim *- SimonaStandaloneSetup SimonaStandaloneSetup *- ExtSimLoader - ExtSimAdapter -- Scheduler + ExtSimAdapter - Scheduler ExtEvDataService -- Scheduler - SecondaryData <|-- EvMovementData + ArrivingEvs --u|> EvData + EvFreeLotsRequest --u|> EvData + DepartingEvsRequest --u|> EvData + SecondaryData <|-- EvData SimonaService <|- ExtEvDataService - ExtEvData <--- ExtEvDataService + ExtEvData --- ExtEvDataService ExtSimAdapterData <--- ExtSimAdapter - ' 1 and n are swapped here due to bug with linetype ortho and labels ExtEvDataService "n" <-> "1" EvcsAgent EvcsAgent --|> ParticipantAgent EvcsAgent *- EvcsModel - ExtSimAdapter -[hidden]> SimonaService } +'Order of classes/interfaces established through hidden connections: + +ExtSimAdapter -[hidden]> SimonaService +sub1 ---[hidden]d> sub2 +ExtLink -[hidden]> ElectricVehicle +ExtSimulation -[hidden]> ExtEvSimulation +ExtSimulation -[hidden]> DataResponseMessageToExt +ProvideArrivingEvs --[hidden]> ExtSimAdapter + @enduml \ No newline at end of file diff --git a/src/test/groovy/edu/ie3/simona/model/participant/StorageModelTest.groovy b/src/test/groovy/edu/ie3/simona/model/participant/StorageModelTest.groovy deleted file mode 100644 index a28a405901..0000000000 --- a/src/test/groovy/edu/ie3/simona/model/participant/StorageModelTest.groovy +++ /dev/null @@ -1,374 +0,0 @@ -/* - * © 2022. TU Dortmund University, - * Institute of Energy Systems, Energy Efficiency and Energy Economics, - * Research group Distribution grid planning and operation - */ - -package edu.ie3.simona.model.participant - -import edu.ie3.datamodel.models.OperationTime -import edu.ie3.datamodel.models.input.NodeInput -import edu.ie3.datamodel.models.input.OperatorInput -import edu.ie3.datamodel.models.input.system.StorageInput -import edu.ie3.datamodel.models.input.system.characteristic.CosPhiFixed -import edu.ie3.datamodel.models.input.system.type.StorageTypeInput -import edu.ie3.datamodel.models.voltagelevels.GermanVoltageLevelUtils -import edu.ie3.simona.ontology.messages.flex.MinMaxFlexibilityMessage.ProvideMinMaxFlexOptions -import edu.ie3.util.TimeUtil -import edu.ie3.util.scala.quantities.Sq -import scala.Option -import spock.lang.Shared -import spock.lang.Specification -import squants.energy.* - -import static edu.ie3.util.quantities.PowerSystemUnits.* -import static tech.units.indriya.quantity.Quantities.getQuantity - -class StorageModelTest extends Specification { - - @Shared - StorageInput inputModel - @Shared - static final Double TOLERANCE = 1e-10 - - def setupSpec() { - def nodeInput = new NodeInput( - UUID.fromString("ad39d0b9-5ad6-4588-8d92-74c7d7de9ace"), - "NodeInput", - OperatorInput.NO_OPERATOR_ASSIGNED, - OperationTime.notLimited(), - getQuantity(1d, PU), - false, - NodeInput.DEFAULT_GEO_POSITION, - GermanVoltageLevelUtils.LV, - -1) - - def typeInput = new StorageTypeInput( - UUID.fromString("fbee4995-24dd-45e4-9c85-7d986fe99ff3"), - "Test_StorageTypeInput", - getQuantity(10000d, EURO), - getQuantity(0.05d, EURO_PER_MEGAWATTHOUR), - getQuantity(100d, KILOWATTHOUR), - getQuantity(13d, KILOVOLTAMPERE), - 0.997, - getQuantity(10d, KILOWATT), - getQuantity(0.03, PU_PER_HOUR), - getQuantity(0.9, PU), - ) - - inputModel = new StorageInput( - UUID.randomUUID(), - "Test_StorageInput", - new OperatorInput(UUID.randomUUID(), "NO_OPERATOR"), - OperationTime.notLimited(), - nodeInput, - CosPhiFixed.CONSTANT_CHARACTERISTIC, - null, - typeInput - ) - } - - def buildStorageModel(Option targetSoc = Option.empty()) { - return StorageModel.apply(inputModel, 1, - TimeUtil.withDefaults.toZonedDateTime("2020-01-01T00:00:00Z"), - TimeUtil.withDefaults.toZonedDateTime("2020-01-01T01:00:00Z"), - 0d, - targetSoc) - } - - def "Calculate flex options"() { - given: - def storageModel = buildStorageModel() - def startTick = 3600L - def data = new StorageModel.StorageRelevantData(startTick + timeDelta) - def oldState = new StorageModel.StorageState( - Sq.create(lastStored.doubleValue(), KilowattHours$.MODULE$), - Sq.create(lastPower.doubleValue(), Kilowatts$.MODULE$), - startTick - ) - - when: - def result = (ProvideMinMaxFlexOptions) storageModel.determineFlexOptions(data, oldState) - - then: - Math.abs(result.ref().toKilowatts() - pRef) < TOLERANCE - Math.abs(result.min().toKilowatts() - pMin) < TOLERANCE - Math.abs(result.max().toKilowatts() - pMax) < TOLERANCE - - where: - lastStored | lastPower | timeDelta || pRef | pMin | pMax - // UNCHANGED STATE - // completely empty - 0 | 0 | 1 || 0 | 0 | 10 - // at a tiny bit above empty - 0.011d | 0 | 1 || 0 | -10 | 10 - // at mid-level charge - 60 | 0 | 1 || 0 | -10 | 10 - // almost fully charged - 99.989d | 0 | 1 || 0 | -10 | 10 - // fully charged - 100 | 0 | 1 || 0 | -10 | 0 - // CHANGED STATE - // discharged to empty - 10 | -9 | 3600 || 0 | 0 | 10 - // almost discharged to lowest allowed charge - 10 | -9 | 3590 || 0 | -10 | 10 - // charged to mid-level charge - 41 | 10 | 3600 || 0 | -10 | 10 - // discharged to mid-level charge - 60 | -9 | 3600 || 0 | -10 | 10 - // almost fully charged - 95.5 | 4.98 | 3600 || 0 | -10 | 10 - // fully charged - 95.5 | 5 | 3600 || 0 | -10 | 0 - } - - def "Calculate flex options with target SOC"() { - given: - def storageModel = buildStorageModel(Option.apply(0.5d)) - def startTick = 3600L - def data = new StorageModel.StorageRelevantData(startTick + 1) - def oldState = new StorageModel.StorageState( - Sq.create(lastStored.doubleValue(), KilowattHours$.MODULE$), - Sq.create(0d, Kilowatts$.MODULE$), - startTick - ) - - when: - def result = (ProvideMinMaxFlexOptions) storageModel.determineFlexOptions(data, oldState) - - then: - Math.abs(result.ref().toKilowatts() - pRef) < TOLERANCE - Math.abs(result.min().toKilowatts() - pMin) < TOLERANCE - Math.abs(result.max().toKilowatts() - pMax) < TOLERANCE - - where: - lastStored || pRef | pMin | pMax - // completely empty - 0 || 10 | 0 | 10 - // below margin of ref power target - 49.9974 || 10 | -10 | 10 - // within margin below ref power target - 49.9976 || 0 | -10 | 10 - // exactly at ref power target - 50 || 0 | -10 | 10 - // within margin above ref power target - 50.0030 || 0 | -10 | 10 - // above margin of ref power target - 50.0031 || -10 | -10 | 10 - // at mid-level charge - 60 || -10 | -10 | 10 - // fully charged - 100 || -10 | -10 | 0 - } - - def "Handle controlled power change"() { - given: - def storageModel = buildStorageModel() - def startTick = 3600L - def data = new StorageModel.StorageRelevantData(startTick + 1) - def oldState = new StorageModel.StorageState( - Sq.create(lastStored.doubleValue(), KilowattHours$.MODULE$), - Sq.create(0d, Kilowatts$.MODULE$), - startTick - ) - - when: - def result = storageModel.handleControlledPowerChange( - data, - oldState, - Sq.create(setPower.doubleValue(), Kilowatts$.MODULE$) - ) - - then: - Math.abs(result._1.chargingPower().toKilowatts() - expPower.doubleValue()) < TOLERANCE - result._1.tick() == startTick + 1 - Math.abs(result._1.storedEnergy().toKilowattHours() - lastStored.doubleValue()) < TOLERANCE - def flexChangeIndication = result._2 - flexChangeIndication.changesAtTick().defined == expScheduled - flexChangeIndication.changesAtTick().map(x -> x == startTick + 1 + expDelta).getOrElse(_ -> true) - flexChangeIndication.changesAtNextActivation() == expActiveNext - - where: - lastStored | setPower || expPower | expActiveNext | expScheduled | expDelta - // no power - 0 | 0 || 0 | false | false | 0 - 50 | 0 || 0 | false | false | 0 - 100 | 0 || 0 | false | false | 0 - // charging on empty - 0 | 1 || 1 | true | true | 100 * 3600 / 0.9 - 0 | 2.5 || 2.5 | true | true | 40 * 3600 / 0.9 - 0 | 5 || 5 | true | true | 20 * 3600 / 0.9 - 0 | 10 || 10 | true | true | 10 * 3600 / 0.9 - // charging on half full - 50 | 5 || 5 | false | true | 10 * 3600 / 0.9 - 50 | 10 || 10 | false | true | 5 * 3600 / 0.9 - // discharging on half full - 50 | -4.5 || -4.5 | false | true | 10 * 3600 - 50 | -9 || -9 | false | true | 5 * 3600 - // discharging on full - 100 | -4.5 || -4.5 | true | true | 20 * 3600 - 100 | -9 || -9 | true | true | 10 * 3600 - } - - def "Handle controlled power change with ref target SOC"() { - given: - def storageModel = buildStorageModel(Option.apply(0.5d)) - def startTick = 3600L - def data = new StorageModel.StorageRelevantData(startTick + 1) - def oldState = new StorageModel.StorageState( - Sq.create(lastStored.doubleValue(), KilowattHours$.MODULE$), - Sq.create(0d, Kilowatts$.MODULE$), - startTick - ) - - when: - def result = storageModel.handleControlledPowerChange( - data, - oldState, - Sq.create(setPower.doubleValue(), Kilowatts$.MODULE$) - ) - - then: - Math.abs(result._1.chargingPower().toKilowatts() - expPower.doubleValue()) < TOLERANCE - result._1.tick() == startTick + 1 - Math.abs(result._1.storedEnergy().toKilowattHours() - lastStored.doubleValue()) < TOLERANCE - def flexChangeIndication = result._2 - flexChangeIndication.changesAtTick().defined == expScheduled - flexChangeIndication.changesAtTick().map(x -> x == startTick + 1 + expDelta).getOrElse(_ -> true) - flexChangeIndication.changesAtNextActivation() == expActiveNext - - where: - lastStored | setPower || expPower | expActiveNext | expScheduled | expDelta - // no power - 0 | 0 || 0 | false | false | 0 - 50 | 0 || 0 | false | false | 0 - 100 | 0 || 0 | false | false | 0 - // charging on empty - 0 | 1 || 1 | true | true | 50 * 3600 / 0.9 - 0 | 2.5 || 2.5 | true | true | 20 * 3600 / 0.9 - 0 | 5 || 5 | true | true | 10 * 3600 / 0.9 - 0 | 10 || 10 | true | true | 5 * 3600 / 0.9 - // charging on target ref - 50 | 5 || 5 | true | true | 10 * 3600 / 0.9 - 50 | 10 || 10 | true | true | 5 * 3600 / 0.9 - // discharging on target ref - 50 | -4.5 || -4.5 | true | true | 10 * 3600 - 50 | -9 || -9 | true | true | 5 * 3600 - // discharging on full - 100 | -4.5 || -4.5 | true | true | 10 * 3600 - 100 | -9 || -9 | true | true | 5 * 3600 - } - - def "Handle the edge case of discharging in tolerance margins"() { - given: - def storageModel = buildStorageModel() - def startTick = 1800L - def data = new StorageModel.StorageRelevantData(startTick + 1) - // margin is at ~ 0.0030864 kWh - def oldState = new StorageModel.StorageState( - Sq.create(0.002d, KilowattHours$.MODULE$), - Sq.create(0d, Kilowatts$.MODULE$), - startTick - ) - - when: - def result = storageModel.handleControlledPowerChange( - data, - oldState, - Sq.create(-5d, Kilowatts$.MODULE$) - ) - - then: - Math.abs(result._1.chargingPower().toKilowatts()) < TOLERANCE - result._1.tick() == startTick + 1 - Math.abs(result._1.storedEnergy().toKilowattHours() - oldState.storedEnergy().toKilowattHours()) < TOLERANCE - def flexChangeIndication = result._2 - !flexChangeIndication.changesAtTick().defined - flexChangeIndication.changesAtNextActivation() - } - - def "Handle the edge case of charging in tolerance margins"() { - given: - def storageModel = buildStorageModel() - def startTick = 1800L - def data = new StorageModel.StorageRelevantData(startTick + 1) - // margin is at ~ 99.9975 kWh - def oldState = new StorageModel.StorageState( - Sq.create(99.999d, KilowattHours$.MODULE$), - Sq.create(0d, Kilowatts$.MODULE$), - startTick - ) - - when: - def result = storageModel.handleControlledPowerChange( - data, - oldState, - Sq.create(9d, Kilowatts$.MODULE$) - ) - - then: - Math.abs(result._1.chargingPower().toKilowatts()) < TOLERANCE - result._1.tick() == startTick + 1 - Math.abs(result._1.storedEnergy().toKilowattHours() - oldState.storedEnergy().toKilowattHours()) < TOLERANCE - def flexChangeIndication = result._2 - !flexChangeIndication.changesAtTick().defined - flexChangeIndication.changesAtNextActivation() - } - - def "Handle the edge case of discharging in positive target margin"() { - given: - def storageModel = buildStorageModel(Option.apply(0.3d)) - def startTick = 1800L - def data = new StorageModel.StorageRelevantData(startTick + 1) - // margin is at ~ 30.0025 kWh - def oldState = new StorageModel.StorageState( - Sq.create(30.0024d, KilowattHours$.MODULE$), - Sq.create(0d, Kilowatts$.MODULE$), - startTick - ) - - when: - def result = storageModel.handleControlledPowerChange( - data, - oldState, - Sq.create(-9d, Kilowatts$.MODULE$) - ) - - then: - Math.abs(result._1.chargingPower().toKilowatts() - (-9d)) < TOLERANCE - result._1.tick() == startTick + 1 - Math.abs(result._1.storedEnergy().toKilowattHours() - oldState.storedEnergy().toKilowattHours()) < TOLERANCE - def flexChangeIndication = result._2 - flexChangeIndication.changesAtTick() == Option.apply(startTick + 1L + 10801L) - flexChangeIndication.changesAtNextActivation() - } - - def "Handle the edge case of charging in negative target margin"() { - given: - def storageModel = buildStorageModel(Option.apply(0.4d)) - def startTick = 1800L - def data = new StorageModel.StorageRelevantData(startTick + 1) - // margin is at ~ 39.9975 kWh - def oldState = new StorageModel.StorageState( - Sq.create(39.998d, KilowattHours$.MODULE$), - Sq.create(0d, Kilowatts$.MODULE$), - startTick - ) - - when: - def result = storageModel.handleControlledPowerChange( - data, - oldState, - Sq.create(5d, Kilowatts$.MODULE$) - ) - - then: - Math.abs(result._1.chargingPower().toKilowatts() - (5d)) < TOLERANCE - result._1.tick() == startTick + 1 - Math.abs(result._1.storedEnergy().toKilowattHours() - oldState.storedEnergy().toKilowattHours()) < TOLERANCE - def flexChangeIndication = result._2 - flexChangeIndication.changesAtTick() == Option.apply(startTick + 1L + 48002L) - flexChangeIndication.changesAtNextActivation() - } -} diff --git a/src/test/scala/edu/ie3/simona/model/participant/StorageModelSpec.scala b/src/test/scala/edu/ie3/simona/model/participant/StorageModelSpec.scala new file mode 100644 index 0000000000..4d6fff7060 --- /dev/null +++ b/src/test/scala/edu/ie3/simona/model/participant/StorageModelSpec.scala @@ -0,0 +1,450 @@ +/* + * © 2022. TU Dortmund University, + * Institute of Energy Systems, Energy Efficiency and Energy Economics, + * Research group Distribution grid planning and operation + */ + +package edu.ie3.simona.model.participant + +import edu.ie3.datamodel.models.OperationTime +import edu.ie3.datamodel.models.input.system.StorageInput +import edu.ie3.datamodel.models.input.system.`type`.StorageTypeInput +import edu.ie3.datamodel.models.input.system.characteristic.CosPhiFixed +import edu.ie3.datamodel.models.input.{NodeInput, OperatorInput} +import edu.ie3.datamodel.models.voltagelevels.GermanVoltageLevelUtils +import edu.ie3.simona.ontology.messages.flex.MinMaxFlexibilityMessage.ProvideMinMaxFlexOptions +import edu.ie3.simona.test.common.UnitSpec +import edu.ie3.util.TimeUtil +import edu.ie3.util.quantities.PowerSystemUnits +import edu.ie3.util.quantities.PowerSystemUnits._ +import edu.ie3.util.scala.quantities.DefaultQuantities.zeroKW +import org.scalatest.matchers.should.Matchers +import squants.energy.{KilowattHours, Kilowatts} +import squants.{Energy, Power} +import tech.units.indriya.quantity.Quantities +import tech.units.indriya.quantity.Quantities.getQuantity + +import java.util.UUID + +class StorageModelSpec extends UnitSpec with Matchers { + + final val inputModel: StorageInput = createStorageInput() + implicit val powerTolerance: Power = Kilowatts(1e-10) + implicit val energyTolerance: Energy = KilowattHours(1e-10) + + def createStorageInput(): StorageInput = { + val nodeInput = new NodeInput( + UUID.fromString("ad39d0b9-5ad6-4588-8d92-74c7d7de9ace"), + "NodeInput", + OperatorInput.NO_OPERATOR_ASSIGNED, + OperationTime.notLimited(), + Quantities.getQuantity(1, PowerSystemUnits.PU), + false, + NodeInput.DEFAULT_GEO_POSITION, + GermanVoltageLevelUtils.LV, + -1, + ) + + val typeInput = new StorageTypeInput( + UUID.fromString("fbee4995-24dd-45e4-9c85-7d986fe99ff3"), + "Test_StorageTypeInput", + Quantities.getQuantity(10000d, EURO), + getQuantity(0.05d, EURO_PER_MEGAWATTHOUR), + Quantities.getQuantity(100d, KILOWATTHOUR), + getQuantity(13d, KILOVOLTAMPERE), + 0.997, + getQuantity(10d, KILOWATT), + getQuantity(0.03, PU_PER_HOUR), + getQuantity(0.9, PU), + ) + + new StorageInput( + UUID.randomUUID(), + "Test_StorageInput", + new OperatorInput(UUID.randomUUID(), "NO_OPERATOR"), + OperationTime.notLimited(), + nodeInput, + CosPhiFixed.CONSTANT_CHARACTERISTIC, + null, + typeInput, + ) + } + + def buildStorageModel( + targetSoc: Option[Double] = Option.empty + ): StorageModel = { + StorageModel.apply( + inputModel, + 1, + TimeUtil.withDefaults.toZonedDateTime("2020-01-01T00:00:00Z"), + TimeUtil.withDefaults.toZonedDateTime("2020-01-01T01:00:00Z"), + 0d, + targetSoc, + ) + } + + "StorageModel" should { + "Calculate flex options" in { + val storageModel = buildStorageModel() + val startTick = 3600L + + val testCases = Table( + ("lastStored", "lastPower", "timeDelta", "pRef", "pMin", "pMax"), + // UNCHANGED STATE + // completely empty + (0.0, 0.0, 1, 0.0, 0.0, 10.0), + // at a tiny bit above empty + (0.011, 0.0, 1, 0.0, -10.0, 10.0), + // at mid-level charge + (60.0, 0.0, 1, 0.0, -10.0, 10.0), + // almost fully charged + (99.989, 0.0, 1, 0.0, -10.0, 10.0), + // fully charged + (100.0, 0.0, 1, 0.0, -10.0, 0.0), + // CHANGED STATE + // discharged to empty + (10.0, -9.0, 3600, 0.0, 0.0, 10.0), + // almost discharged to lowest allowed charge + (10.0, -9.0, 3590, 0.0, -10.0, 10.0), + // charged to mid-level charge + (41.0, 10.0, 3600, 0.0, -10.0, 10.0), + // discharged to mid-level charge + (60.0, -9.0, 3600, 0.0, -10.0, 10.0), + // almost fully charged + (95.5, 4.98, 3600, 0.0, -10.0, 10.0), + // fully charged + (95.5, 5.0, 3600, 0.0, -10.0, 0.0), + ) + + forAll(testCases) { + ( + lastStored: Double, + lastPower: Double, + timeDelta: Int, + pRef: Double, + pMin: Double, + pMax: Double, + ) => + val data = StorageModel.StorageRelevantData(startTick + timeDelta) + val oldState = StorageModel.StorageState( + KilowattHours(lastStored), + Kilowatts(lastPower), + startTick, + ) + + storageModel + .determineFlexOptions(data, oldState) match { + case result: ProvideMinMaxFlexOptions => + result.ref should approximate(Kilowatts(pRef)) + result.min should approximate(Kilowatts(pMin)) + result.max should approximate(Kilowatts(pMax)) + case _ => + fail("Expected result of type ProvideMinMaxFlexOptions") + } + } + } + "Calculate flex options with target SOC" in { + val storageModel = buildStorageModel(Some(0.5d)) + val startTick = 3600L + val data = StorageModel.StorageRelevantData(startTick + 1) + + val testCases = Table( + ("lastStored", "pRef", "pMin", "pMax"), + // completely empty + (0.0, 10.0, 0.0, 10.0), + // below margin of ref power target + (49.9974, 10.0, -10.0, 10.0), + // within margin below ref power target + (49.9976, 0.0, -10.0, 10.0), + // exactly at ref power target + (50.0, 0.0, -10.0, 10.0), + // within margin above ref power target + (50.0030, 0.0, -10.0, 10.0), + // above margin of ref power target + (50.0031, -10.0, -10.0, 10.0), + // at mid-level charge + (60.0, -10.0, -10.0, 10.0), + // fully charged + (100.0, -10.0, -10.0, 0.0), + ) + + forAll(testCases) { + (lastStored: Double, pRef: Double, pMin: Double, pMax: Double) => + val oldState = StorageModel.StorageState( + KilowattHours(lastStored), + zeroKW, + startTick, + ) + + storageModel + .determineFlexOptions(data, oldState) match { + case result: ProvideMinMaxFlexOptions => + result.ref should approximate(Kilowatts(pRef)) + result.min should approximate(Kilowatts(pMin)) + result.max should approximate(Kilowatts(pMax)) + case _ => + fail("Expected result of type ProvideMinMaxFlexOptions") + } + } + } + + "Handle controlled power change" in { + val storageModel = buildStorageModel() + val startTick = 3600L + val data = StorageModel.StorageRelevantData(startTick + 1) + val testCases = Table( + ( + "lastStored", + "setPower", + "expPower", + "expActiveNext", + "expScheduled", + "expDelta", + ), + // no power + (0.0, 0.0, 0.0, false, false, 0.0), + (50.0, 0.0, 0.0, false, false, 0.0), + (100.0, 0.0, 0.0, false, false, 0.0), + // charging on empty + (0.0, 1.0, 1.0, true, true, 100 * 3600 / 0.9), + (0.0, 2.5, 2.5, true, true, 40 * 3600 / 0.9), + (0.0, 5.0, 5.0, true, true, 20 * 3600 / 0.9), + (0.0, 10.0, 10.0, true, true, 10 * 3600 / 0.9), + // charging on half full + (50.0, 5.0, 5.0, false, true, 10 * 3600 / 0.9), + (50.0, 10.0, 10.0, false, true, 5 * 3600 / 0.9), + // discharging on half full + (50.0, -4.5, -4.5, false, true, 10 * 3600.0), + (50.0, -9.0, -9.0, false, true, 5 * 3600.0), + // discharging on full + (100.0, -4.5, -4.5, true, true, 20 * 3600.0), + (100.0, -9.0, -9.0, true, true, 10 * 3600.0), + ) + + forAll(testCases) { + ( + lastStored: Double, + setPower: Double, + expPower: Double, + expActiveNext: Boolean, + expScheduled: Boolean, + expDelta: Double, + ) => + val oldState = StorageModel.StorageState( + KilowattHours(lastStored), + zeroKW, + startTick, + ) + + val (newState, flexChangeIndication) = + storageModel.handleControlledPowerChange( + data, + oldState, + Kilowatts(setPower), + ) + + newState.chargingPower should approximate(Kilowatts(expPower)) + newState.tick shouldBe (startTick + 1) + newState.storedEnergy should approximate(KilowattHours(lastStored)) + + flexChangeIndication.changesAtTick.isDefined shouldBe expScheduled + flexChangeIndication.changesAtTick.forall( + _ == (startTick + 1 + expDelta) + ) shouldBe true + flexChangeIndication.changesAtNextActivation shouldBe expActiveNext + } + } + + "Handle controlled power change with ref target SOC" in { + val storageModel = buildStorageModel(Some(0.5d)) + val startTick = 3600L + val data = StorageModel.StorageRelevantData(startTick + 1) + + val testCases = Table( + ( + "lastStored", + "setPower", + "expPower", + "expActiveNext", + "expScheduled", + "expDelta", + ), + // no power + (0.0, 0.0, 0.0, false, false, 0.0), + (50.0, 0.0, 0.0, false, false, 0.0), + (100.0, 0.0, 0.0, false, false, 0.0), + // charging on empty + (0.0, 1.0, 1.0, true, true, 50 * 3600 / 0.9), + (0.0, 2.5, 2.5, true, true, 20 * 3600 / 0.9), + (0.0, 5.0, 5.0, true, true, 10 * 3600 / 0.9), + (0.0, 10.0, 10.0, true, true, 5 * 3600 / 0.9), + // charging on target ref + (50.0, 5.0, 5.0, true, true, 10 * 3600 / 0.9), + (50.0, 10.0, 10.0, true, true, 5 * 3600 / 0.9), + // discharging on target ref + (50.0, -4.5, -4.5, true, true, 10 * 3600.0), + (50.0, -9.0, -9.0, true, true, 5 * 3600.0), + // discharging on full + (100.0, -4.5, -4.5, true, true, 10 * 3600.0), + (100.0, -9.0, -9.0, true, true, 5 * 3600.0), + ) + + forAll(testCases) { + ( + lastStored: Double, + setPower: Double, + expPower: Double, + expActiveNext: Boolean, + expScheduled: Boolean, + expDelta: Double, + ) => + val oldState = StorageModel.StorageState( + KilowattHours(lastStored), + zeroKW, + startTick, + ) + + val (newState, flexChangeIndication) = + storageModel.handleControlledPowerChange( + data, + oldState, + Kilowatts(setPower), + ) + + newState.chargingPower should approximate(Kilowatts(expPower)) + newState.tick shouldBe (startTick + 1) + newState.storedEnergy should approximate(KilowattHours(lastStored)) + + flexChangeIndication.changesAtTick.isDefined shouldBe expScheduled + flexChangeIndication.changesAtTick.forall( + _ == (startTick + 1 + expDelta) + ) shouldBe true + flexChangeIndication.changesAtNextActivation shouldBe expActiveNext + } + } + + "Handle the edge case of discharging in tolerance margins" in { + val storageModel = buildStorageModel() + val startTick = 1800L + val data = StorageModel.StorageRelevantData(startTick + 1) + // margin is at ~ 0.0030864 kWh + val oldState = StorageModel.StorageState( + KilowattHours(0.002d), + zeroKW, + startTick, + ) + + val (newState, flexChangeIndication) = + storageModel.handleControlledPowerChange( + data, + oldState, + Kilowatts(-5d), + ) + + newState.chargingPower should approximate( + zeroKW + ) + newState.tick shouldBe (startTick + 1) + newState.storedEnergy should approximate( + oldState.storedEnergy + ) + + flexChangeIndication.changesAtTick.isDefined shouldBe false + flexChangeIndication.changesAtNextActivation shouldBe true + } + + "Handle the edge case of charging in tolerance margins" in { + val storageModel = buildStorageModel() + val startTick = 1800L + val data = StorageModel.StorageRelevantData(startTick + 1) + // margin is at ~ 99.9975 kWh + val oldState = StorageModel.StorageState( + KilowattHours(99.999d), + zeroKW, + startTick, + ) + + val (newState, flexChangeIndication) = + storageModel.handleControlledPowerChange( + data, + oldState, + Kilowatts(9d), + ) + + newState.chargingPower should approximate( + zeroKW + ) + newState.tick shouldBe (startTick + 1) + newState.storedEnergy should approximate( + oldState.storedEnergy + ) + + flexChangeIndication.changesAtTick.isDefined shouldBe false + flexChangeIndication.changesAtNextActivation shouldBe true + } + "Handle the edge case of discharging in positive target margin" in { + val storageModel = buildStorageModel(Some(0.3d)) + val startTick = 1800L + val data = StorageModel.StorageRelevantData(startTick + 1) + // margin is at ~ 30.0025 kWh + val oldState = StorageModel.StorageState( + KilowattHours(30.0024d), + zeroKW, + startTick, + ) + + val (newState, flexChangeIndication) = + storageModel.handleControlledPowerChange( + data, + oldState, + Kilowatts(-9d), + ) + + newState.chargingPower should approximate( + Kilowatts(-9d) + ) + newState.tick shouldBe (startTick + 1) + newState.storedEnergy should approximate( + oldState.storedEnergy + ) + flexChangeIndication.changesAtTick should be( + Some(startTick + 1L + 10801L) + ) + flexChangeIndication.changesAtNextActivation should be( + true + ) + } + "Handle the edge case of charging in negative target margin" in { + val storageModel = buildStorageModel(Some(0.4d)) + val startTick = 1800L + val data = StorageModel.StorageRelevantData(startTick + 1) + // margin is at ~ 39.9975 kWh + val oldState = StorageModel.StorageState( + KilowattHours(39.998d), + zeroKW, + startTick, + ) + + val (newState, flexChangeIndication) = + storageModel.handleControlledPowerChange( + data, + oldState, + Kilowatts(5d), + ) + + newState.chargingPower should approximate( + Kilowatts(5d) + ) + newState.tick shouldBe (startTick + 1) + newState.storedEnergy should approximate( + oldState.storedEnergy + ) + flexChangeIndication.changesAtTick should be( + Some(startTick + 1L + 48002L) + ) + flexChangeIndication.changesAtNextActivation should be( + true + ) + } + } +}