Skip to content

Commit

Permalink
Merge branch 'dev' into df/#930-refactor-handleInfeed
Browse files Browse the repository at this point in the history
# Conflicts:
#	src/main/scala/edu/ie3/simona/model/participant/HpModel.scala
#	src/main/scala/edu/ie3/simona/model/thermal/ThermalGrid.scala
#	src/test/scala/edu/ie3/simona/model/thermal/ThermalGridWithHouseOnlySpec.scala
#	src/test/scala/edu/ie3/simona/model/thermal/ThermalGridWithStorageOnlySpec.scala
  • Loading branch information
danielfeismann committed Nov 26, 2024
2 parents 1586bd3 + e420a54 commit 88c6e3a
Show file tree
Hide file tree
Showing 8 changed files with 128 additions and 137 deletions.
1 change: 1 addition & 0 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -107,6 +107,7 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0
- Removing logs in `logs/simona` [#1017](https://github.com/ie3-institute/simona/issues/1017)
- Fix implausible test cases of HpModelSpec [#1042](https://github.com/ie3-institute/simona/issues/1042)
- Refactoring to only use 'lastHpState' and 'relevantData' for 'ThermalGrid' calculations [#916](https://github.com/ie3-institute/simona/issues/916)
- Refactor thermal calcRelevantData [#1051](https://github.com/ie3-institute/simona/issues/1051)

### Fixed
- Fix rendering of references in documentation [#505](https://github.com/ie3-institute/simona/issues/505)
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -250,10 +250,9 @@ final case class HpModel(
/* Push thermal energy to the thermal grid and get its updated state in return */
val (thermalGridState, maybeThreshold) =
thermalGrid.updateState(
relevantData.currentTick,
relevantData,
lastState.thermalGridState,
lastState.ambientTemperature.getOrElse(relevantData.ambientTemperature),
relevantData.ambientTemperature,
isRunning,
newThermalPower,
demandWrapper,
Expand Down
71 changes: 22 additions & 49 deletions src/main/scala/edu/ie3/simona/model/thermal/ThermalGrid.scala
Original file line number Diff line number Diff line change
Expand Up @@ -66,12 +66,11 @@ final case class ThermalGrid(
case Some((thermalHouse, lastHouseState)) =>
val (updatedHouseState, _) =
thermalHouse.determineState(
relevantData.currentTick,
relevantData,
lastHouseState,
lastHpState.ambientTemperature.getOrElse(
relevantData.ambientTemperature
),
relevantData.ambientTemperature,
lastHouseState.qDot,
)
if (
Expand All @@ -80,8 +79,7 @@ final case class ThermalGrid(
) {
(
thermalHouse.energyDemand(
relevantData.currentTick,
relevantData.ambientTemperature,
relevantData,
updatedHouseState,
),
Some(updatedHouseState),
Expand Down Expand Up @@ -147,14 +145,12 @@ final case class ThermalGrid(

/** Update the current state of the grid
*
* @param tick
* Instance in time
* @param relevantData
* data of heat pump including state of the heat pump
* @param state
* Currently applicable state
* @param lastAmbientTemperature
* Ambient temperature valid up until (not including) the current tick
* @param ambientTemperature
* Current ambient temperature
* @param isRunning
* determines whether the heat pump is running or not
* @param qDot
Expand All @@ -165,41 +161,29 @@ final case class ThermalGrid(
* The updated state of the grid
*/
def updateState(
tick: Long,
relevantData: HpRelevantData,
state: ThermalGridState,
lastAmbientTemperature: Temperature,
ambientTemperature: Temperature,
isRunning: Boolean,
qDot: Power,
thermalDemands: ThermalDemandWrapper,
): (ThermalGridState, Option[ThermalThreshold]) = if (qDot > zeroKW)
handleInfeed(
tick,
lastAmbientTemperature,
ambientTemperature,
state,
isRunning,
qDot,
thermalDemands,
)
handleInfeed(relevantData, lastAmbientTemperature, state, isRunning, qDot, thermalDemands)
else
handleConsumption(
tick,
relevantData,
lastAmbientTemperature,
ambientTemperature,
state,
qDot,
)

/** Handles the case, when a grid has infeed. Depending which entity has some
* heat demand the house or the storage will be heated up / filled up.
*
* @param tick
* Current tick
* @param relevantData
* data of heat pump including state of the heat pump
* @param lastAmbientTemperature
* Ambient temperature valid up until (not including) the current tick
* @param ambientTemperature
* Current ambient temperature
* @param state
* Current state of the houses
* @param isRunning
Expand All @@ -212,9 +196,8 @@ final case class ThermalGrid(
* Updated thermal grid state
*/
private def handleInfeed(
tick: Long,
relevantData: HpRelevantData,
lastAmbientTemperature: Temperature,
ambientTemperature: Temperature,
state: ThermalGridState,
isRunning: Boolean,
qDot: Power,
Expand Down Expand Up @@ -514,10 +497,9 @@ final case class ThermalGrid(
) {
val (fullHouseState, maybeFullHouseThreshold) =
thermalHouse.determineState(
tick,
relevantData,
lastHouseState,
lastAmbientTemperature,
ambientTemperature,
zeroKW,
)
(Some(fullHouseState), maybeFullHouseThreshold, qDot)
Expand Down Expand Up @@ -583,12 +565,10 @@ final case class ThermalGrid(

/** Handle consumption (or no infeed) from thermal grid
*
* @param tick
* Current tick
* @param relevantData
* data of heat pump including state of the heat pump
* @param lastAmbientTemperature
* Ambient temperature valid up until (not including) the current tick
* @param ambientTemperature
* Current ambient temperature
* @param state
* Current state of the houses
* @param qDot
Expand All @@ -597,39 +577,36 @@ final case class ThermalGrid(
* Updated thermal grid state
*/
private def handleConsumption(
tick: Long,
relevantData: HpRelevantData,
lastAmbientTemperature: Temperature,
ambientTemperature: Temperature,
state: ThermalGridState,
qDot: Power,
): (ThermalGridState, Option[ThermalThreshold]) = {
/* House will be left with no influx in all cases. Determine if and when a threshold is reached */
val maybeUpdatedHouseState =
house.zip(state.houseState).map { case (house, houseState) =>
house.determineState(
tick,
relevantData,
houseState,
lastAmbientTemperature,
ambientTemperature,
zeroMW,
)
}

/* Update the state of the storage */
val maybeUpdatedStorageState =
storage.zip(state.storageState).map { case (storage, storageState) =>
storage.updateState(tick, qDot, storageState)
storage.updateState(relevantData.currentTick, qDot, storageState)
}

val (revisedHouseState, revisedStorageState) =
reviseInfeedFromStorage(
tick,
relevantData,
maybeUpdatedHouseState,
maybeUpdatedStorageState,
state.houseState,
state.storageState,
lastAmbientTemperature,
ambientTemperature,
qDot,
)

Expand All @@ -652,8 +629,8 @@ final case class ThermalGrid(
* is no infeed from external and</li> <li>the storage is not empty
* itself</li> </ul>
*
* @param tick
* The current tick
* @param relevantData
* data of heat pump including state of the heat pump
* @param maybeHouseState
* Optional thermal house state
* @param maybeStorageState
Expand All @@ -664,23 +641,20 @@ final case class ThermalGrid(
* Previous thermal storage state before a first update was performed
* @param lastAmbientTemperature
* Ambient temperature valid up until (not including) the current tick
* @param ambientTemperature
* Current ambient temperature
* @param qDot
* Thermal influx
* @return
* Options to revised thermal house and storage state
*/
def reviseInfeedFromStorage(
tick: Long,
relevantData: HpRelevantData,
maybeHouseState: Option[(ThermalHouseState, Option[ThermalThreshold])],
maybeStorageState: Option[
(ThermalStorageState, Option[ThermalThreshold])
],
formerHouseState: Option[ThermalHouseState],
formerStorageState: Option[ThermalStorageState],
lastAmbientTemperature: Temperature,
ambientTemperature: Temperature,
qDot: Power,
): (
Option[(ThermalHouseState, Option[ThermalThreshold])],
Expand All @@ -698,7 +672,7 @@ final case class ThermalGrid(
) && !thermalStorage.isEmpty(storageState.storedEnergy) =>
/* Storage is meant to heat the house only, if there is no infeed from external (+/- 10 W) and the house is cold */
val revisedStorageState = thermalStorage.updateState(
tick,
relevantData.currentTick,
thermalStorage.getChargingPower * -1,
formerStorageState.getOrElse(
throw new InconsistentStateException(
Expand All @@ -707,14 +681,13 @@ final case class ThermalGrid(
),
)
val revisedHouseState = thermalHouse.determineState(
tick,
relevantData,
formerHouseState.getOrElse(
throw new InconsistentStateException(
"Impossible to find no house state"
)
),
lastAmbientTemperature,
ambientTemperature,
thermalStorage.getChargingPower,
)
(Some(revisedHouseState), Some(revisedStorageState))
Expand Down
34 changes: 17 additions & 17 deletions src/main/scala/edu/ie3/simona/model/thermal/ThermalHouse.scala
Original file line number Diff line number Diff line change
Expand Up @@ -12,6 +12,7 @@ import edu.ie3.datamodel.models.input.thermal.{
ThermalBusInput,
ThermalHouseInput,
}
import edu.ie3.simona.model.participant.HpModel.HpRelevantData
import edu.ie3.simona.model.thermal.ThermalGrid.ThermalEnergyDemand
import edu.ie3.simona.model.thermal.ThermalHouse.ThermalHouseThreshold.{
HouseTemperatureLowerBoundaryReached,
Expand Down Expand Up @@ -82,27 +83,24 @@ final case class ThermalHouse(
* determining the thermal demand, a change in external infeed will take
* place.
*
* @param tick
* Questionable tick
* @param ambientTemperature
* Ambient temperature in the instance in question
* @param relevantData
* data of heat pump including state of the heat pump
* @param state
* most recent state, that is valid for this model
* @return
* the needed energy in the questioned tick
*/
def energyDemand(
tick: Long,
ambientTemperature: Temperature,
relevantData: HpRelevantData,
state: ThermalHouseState,
): ThermalEnergyDemand = {
/* Calculate the inner temperature of the house, at the questioned instance in time */
val duration = Seconds(tick - state.tick)
val duration = Seconds(relevantData.currentTick - state.tick)
val currentInnerTemp = newInnerTemperature(
state.qDot,
duration,
state.innerTemperature,
ambientTemperature,
relevantData.ambientTemperature,
)

/* Determine, which temperature boundary triggers a needed energy to reach the temperature constraints */
Expand Down Expand Up @@ -219,27 +217,24 @@ final case class ThermalHouse(

/** Update the current state of the house
*
* @param tick
* Current instance in time
* @param relevantData
* data of heat pump including state of the heat pump
* @param state
* Currently applicable state
* @param lastAmbientTemperature
* Ambient temperature valid up until (not including) the current tick
* @param ambientTemperature
* Current ambient temperature
* @param qDot
* New thermal influx
* @return
* Updated state and the tick in which the next threshold is reached
*/
def determineState(
tick: Long,
relevantData: HpRelevantData,
state: ThermalHouseState,
lastAmbientTemperature: Temperature,
ambientTemperature: Temperature,
qDot: Power,
): (ThermalHouseState, Option[ThermalThreshold]) = {
val duration = Seconds(tick - state.tick)
val duration = Seconds(relevantData.currentTick - state.tick)
val updatedInnerTemperature = newInnerTemperature(
state.qDot,
duration,
Expand All @@ -249,11 +244,16 @@ final case class ThermalHouse(

/* Calculate the next given threshold */
val threshold =
nextThreshold(tick, qDot, updatedInnerTemperature, ambientTemperature)
nextThreshold(
relevantData.currentTick,
qDot,
updatedInnerTemperature,
relevantData.ambientTemperature,
)

(
state.copy(
tick = tick,
tick = relevantData.currentTick,
innerTemperature = updatedInnerTemperature,
qDot = qDot,
),
Expand Down
Loading

0 comments on commit 88c6e3a

Please sign in to comment.