123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356 |
- # -*- coding: utf-8 -*-
- from operator import attrgetter
- import numpy as np
- from fastapi import HTTPException
- from loguru import logger
- from app.api.errors.iot import MissingIOTDataError
- from app.controllers.equipment.controller import EquipmentController
- from app.models.domain.devices import ACATVAInstructionsRequest, ACATVAInstructionsRequestV2
- from app.schemas.equipment import VAVBox, FCU
- from app.schemas.instructions import ACATVAInstructions
- from app.schemas.sapce_weight import SpaceWeight
- from app.schemas.space import SpaceATVA
- from app.services.transfer import Season
- from app.utils.date import get_time_str
- class VAVController(EquipmentController):
- def __init__(self, equipment: VAVBox):
- super(VAVController, self).__init__()
- self.equipment = equipment
- async def get_strategy(self):
- strategy = "Plan A"
- for space in self.equipment.spaces:
- for eq in space.equipment:
- if isinstance(eq, FCU):
- strategy = "Plan B"
- break
- return strategy
- async def build_virtual_temperature(self) -> tuple[float, float]:
- target_list, realtime_list = [], []
- buffer_list = []
- strategy = await self.get_strategy()
- for space in self.equipment.spaces:
- if not np.isnan(space.temperature_target):
- target_list.append(space.temperature_target)
- realtime_list.append(space.realtime_temperature)
- if strategy == "Plan B":
- for eq in space.equipment:
- if isinstance(eq, FCU):
- buffer = (4 - eq.air_valve_speed) / 4
- buffer_list.append(buffer)
- break
- total_target = buffer_list + target_list
- total_realtime = buffer_list + realtime_list
- if total_target and total_realtime:
- target_result = np.array(total_target).sum() / len(target_list)
- realtime_result = np.array(total_realtime).sum() / len(realtime_list)
- self.equipment.setting_temperature = target_result
- else:
- target_result, realtime_result = np.NAN, np.NAN
- return target_result, realtime_result
- async def get_supply_air_flow_set(
- self, temperature_set: float, temperature_realtime: float
- ) -> float:
- if not (temperature_set and temperature_realtime):
- supply_air_flow_set = 0.0
- else:
- temperature_supply = self.equipment.supply_air_temperature
- if np.isnan(temperature_supply):
- temperature_supply = 19.0
- try:
- ratio = abs(
- 1
- + (temperature_realtime - temperature_set)
- / (temperature_set - temperature_supply)
- )
- except ZeroDivisionError:
- ratio = 1
- supply_air_flow_set = self.equipment.supply_air_flow * ratio
- supply_air_flow_set = max(
- self.equipment.supply_air_flow_lower_limit, supply_air_flow_set
- )
- supply_air_flow_set = min(
- self.equipment.supply_air_flow_upper_limit, supply_air_flow_set
- )
- self.equipment.supply_air_flow_set = supply_air_flow_set
- self.equipment.virtual_target_temperature = temperature_set
- self.equipment.virtual_realtime_temperature = temperature_realtime
- return supply_air_flow_set
- async def run(self):
- temperature_set, temperature_realtime = await self.build_virtual_temperature()
- await self.get_supply_air_flow_set(temperature_set, temperature_realtime)
- self.equipment.running_status = True
- def get_results(self):
- return self.equipment
- class VAVControllerV2(VAVController):
- def __init__(
- self,
- equipment: VAVBox,
- weights: list[SpaceWeight] | None = None,
- season: Season | None = None,
- ):
- super(VAVControllerV2, self).__init__(equipment)
- self.weights = weights
- self.season = season
- def get_valid_spaces(self) -> list[SpaceATVA]:
- valid_spaces = list()
- for sp in self.equipment.spaces:
- if sp.realtime_temperature and sp.temperature_target:
- sp.diff = sp.temperature_target - sp.realtime_temperature
- valid_spaces.append(sp)
- return valid_spaces
- async def build_virtual_temperature(self) -> None:
- valid_spaces = []
- weights = []
- for sp in self.equipment.spaces:
- if sp.realtime_temperature > 0.0 and sp.temperature_target > 0.0:
- valid_spaces.append(sp)
- for weight in self.weights:
- if weight.space_id == sp.id:
- weights.append(weight)
- if valid_spaces:
- weights = sorted(weights, key=lambda x: x.temporary_weight_update_time)
- if weights[-1].temporary_weight_update_time > get_time_str(
- 60 * 60 * 2, flag="ago"
- ):
- # If someone has submitted a feedback in past two hours, meet the need.
- weight_dic = {weight.space_id: 0.0 for weight in weights}
- weight_dic.update({weights[-1].space_id: weights[-1].temporary_weight})
- else:
- weight_dic = {
- weight.space_id: weight.default_weight for weight in weights
- }
- total_weight_value = 0.0
- for v in weight_dic.values():
- total_weight_value += v
- if total_weight_value > 0:
- weight_dic = {
- k: v / total_weight_value for k, v in weight_dic.items()
- }
- else:
- weight_dic.update({list(weight_dic.keys())[0]: 1.0})
- try:
- virtual_target, virtual_realtime = 0.0, 0.0
- for sp in valid_spaces:
- virtual_target += sp.temperature_target * weight_dic.get(sp.id)
- virtual_realtime += sp.realtime_temperature * weight_dic.get(sp.id)
- except KeyError:
- logger.error(f"{self.equipment.id} has wrong vav-space relation")
- raise HTTPException(
- status_code=404, detail="This VAV box has wrong eq-sp relation"
- )
- self.equipment.virtual_target_temperature = virtual_target
- self.equipment.virtual_realtime_temperature = virtual_realtime
- else:
- self.equipment.virtual_target_temperature = np.NAN
- self.equipment.virtual_realtime_temperature = np.NAN
- async def rectify(self) -> tuple[float, float]:
- bad_spaces = list()
- valid_spaces = self.get_valid_spaces()
- for sp in valid_spaces:
- if self.season == Season.heating:
- if sp.realtime_temperature > max(
- 26.0, sp.temperature_target
- ) or sp.realtime_temperature < min(20.0, sp.temperature_target):
- if sp.temperature_target > 0.0:
- bad_spaces.append(sp)
- elif self.season == Season.cooling:
- if sp.realtime_temperature > max(
- 27.0, sp.temperature_target
- ) or sp.realtime_temperature < min(22.0, sp.temperature_target):
- if sp.temperature_target > 0.0:
- bad_spaces.append(sp)
- if bad_spaces:
- virtual_diff = self.equipment.virtual_target_temperature - self.equipment.virtual_realtime_temperature
- if self.season == Season.cooling:
- bad_spaces = sorted(bad_spaces, key=attrgetter("diff"))
- worst = bad_spaces[0]
- if worst.diff * virtual_diff >= 0:
- if abs(worst.diff) > abs(virtual_diff):
- self.equipment.virtual_target_temperature = worst.temperature_target
- self.equipment.virtual_realtime_temperature = worst.realtime_temperature
- else:
- if worst.diff < 0:
- self.equipment.virtual_target_temperature = min(22.0, worst.temperature_target) + 0.5
- else:
- self.equipment.virtual_target_temperature = max(26.0, worst.temperature_target) - 0.5
- self.equipment.virtual_realtime_temperature = worst.realtime_temperature
- elif self.season == Season.heating:
- bad_spaces = sorted(bad_spaces, key=attrgetter("diff"), reverse=True)
- worst = bad_spaces[0]
- if worst.diff * virtual_diff >= 0:
- if abs(worst.diff) > abs(virtual_diff):
- self.equipment.virtual_target_temperature = worst.temperature_target
- self.equipment.virtual_realtime_temperature = worst.realtime_temperature
- else:
- if worst.diff > 0:
- self.equipment.virtual_target_temperature = max(26.0, worst.temperature_target) - 0.5
- else:
- self.equipment.virtual_target_temperature = min(20.0, worst.temperature_target) + 0.5
- self.equipment.virtual_realtime_temperature = worst.realtime_temperature
- return (
- self.equipment.virtual_target_temperature,
- self.equipment.virtual_realtime_temperature,
- )
- async def run(self) -> None:
- try:
- await self.build_virtual_temperature()
- temperature_set, temperature_realtime = await self.rectify()
- await self.get_supply_air_flow_set(temperature_set, temperature_realtime)
- self.equipment.running_status = True
- except TypeError:
- raise MissingIOTDataError
- class VAVControllerV3(VAVControllerV2):
- def __init__(self, vav_params: VAVBox, season: Season):
- super(VAVControllerV3, self).__init__(vav_params)
- self.season = season
- async def build_virtual_temperature(self) -> None:
- valid_spaces = self.get_valid_spaces()
- if not valid_spaces:
- raise MissingIOTDataError
- else:
- sorted_spaces = sorted(valid_spaces, key=lambda x: x.vav_temporary_update_time)
- if sorted_spaces[-1].vav_temporary_update_time > get_time_str(60 * 60 * 2, flag="ago"):
- virtual_realtime = sorted_spaces[-1].realtime_temperature
- virtual_target = sorted_spaces[-1].temperature_target
- else:
- virtual_realtime, virtual_target = 0.0, 0.0
- total_weight = 0.0
- for sp in valid_spaces:
- temp_weight = sp.vav_default_weight
- virtual_realtime += sp.realtime_temperature * temp_weight
- virtual_target += sp.temperature_target * temp_weight
- total_weight += temp_weight
- if total_weight == 0:
- for sp in valid_spaces:
- virtual_realtime += sp.realtime_temperature
- virtual_target += sp.temperature_target
- virtual_realtime /= len(valid_spaces)
- virtual_target /= len(valid_spaces)
- else:
- virtual_realtime /= total_weight
- virtual_target /= total_weight
- self.equipment.virtual_realtime_temperature = virtual_realtime
- self.equipment.virtual_target_temperature = virtual_target
- class VAVControllerV4(VAVControllerV3):
- def __init__(self, vav_params: VAVBox, season: Season, return_air_temp: float):
- super().__init__(vav_params, season)
- self.return_air_temp = return_air_temp
- def get_next_temp_set(self, virtual_realtime_temp: float, virtual_target_temp: float) -> float:
- if not (virtual_realtime_temp and virtual_target_temp):
- next_temp_set = np.NAN
- else:
- next_temp_set = virtual_target_temp + self.return_air_temp - virtual_realtime_temp
- self.equipment.setting_temperature = next_temp_set
- return next_temp_set
- async def run(self) -> None:
- try:
- await self.build_virtual_temperature()
- temperature_set, temperature_realtime = await self.rectify()
- self.get_next_temp_set(temperature_realtime, temperature_set)
- self.equipment.running_status = True
- except TypeError:
- raise MissingIOTDataError
- async def build_acatva_instructions(params: ACATVAInstructionsRequest) -> ACATVAInstructions:
- space_params = []
- for sp in params.spaces:
- temp_sp = SpaceATVA(**sp.dict())
- space_params.append(temp_sp)
- if params.supply_air_temperature:
- supply_air_temperature = params.supply_air_temperature
- else:
- supply_air_temperature = params.acatah_supply_air_temperature
- vav_params = VAVBox(
- spaces=space_params,
- supply_air_temperature=supply_air_temperature,
- supply_air_flow=params.supply_air_flow,
- supply_air_flow_lower_limit=params.supply_air_flow_lower_limit,
- supply_air_flow_upper_limit=params.supply_air_flow_upper_limit,
- )
- controller = VAVControllerV3(vav_params=vav_params, season=Season(params.season))
- await controller.run()
- regulated_vav = controller.get_results()
- instructions = ACATVAInstructions(
- supply_air_flow_set=regulated_vav.supply_air_flow_set,
- virtual_realtime_temperature=regulated_vav.virtual_realtime_temperature,
- virtual_temperature_target_set=regulated_vav.virtual_target_temperature,
- )
- return instructions
- async def build_acatva_instructions_for_jm(params: ACATVAInstructionsRequestV2) -> dict:
- # Control logic for Jiaming.
- space_params = []
- for sp in params.spaces:
- temp_sp = SpaceATVA(**sp.dict())
- space_params.append(temp_sp)
- vav_params = VAVBox(spaces=space_params)
- controller = VAVControllerV4(
- vav_params=vav_params,
- season=Season(params.season),
- return_air_temp=params.return_air_temperature,
- )
- await controller.run()
- regulated_vav = controller.get_results()
- next_temp_set = regulated_vav.setting_temperature
- if next_temp_set:
- if np.isnan(next_temp_set):
- next_temp_set = -1.0
- else:
- next_temp_set = -1.0
- instructions = {
- 'temperature_target_set': next_temp_set,
- 'virtual_target_temperature': regulated_vav.virtual_target_temperature,
- 'virtual_realtime_temperature': regulated_vav.virtual_realtime_temperature
- }
- return instructions
|