# Copyright (c) 2019-2020, RTE (https://www.rte-france.com)
# See AUTHORS.txt
# This Source Code Form is subject to the terms of the Mozilla Public License, version 2.0.
# If a copy of the Mozilla Public License, version 2.0 was not distributed with this file,
# you can obtain one at http://mozilla.org/MPL/2.0/.
# SPDX-License-Identifier: MPL-2.0
# This file is part of Grid2Op, Grid2Op a testbed platform to model sequential decision making in power systems.
from typing import Optional, Dict, Literal
import warnings
from grid2op.Exceptions import AmbiguousAction
from grid2op.Action.baseAction import BaseAction
[docs]class VoltageOnlyAction(BaseAction):
"""
INTERNAL
.. warning:: /!\\\\ Internal, do not use unless you know what you are doing /!\\\\
This is the main class used by the voltage controller.
This class is here to serve as a base class for the controller of the voltages (if any). It allows to perform
only modification of the generator voltage set point.
Only action of type "injection" are supported, and only setting "prod_v" keyword.
"""
authorized_keys = {"injection"}
attr_list_vect = ["prod_v"]
attr_list_set = set(attr_list_vect)
_shunt_added = False
_first_init = True
[docs] def __init__(self, _names_chronics_to_backend: Optional[Dict[Literal["loads", "prods", "lines"], Dict[str, str]]]=None):
"""
See the definition of :func:`BaseAction.__init__` and of :class:`BaseAction` for more information. Nothing more is done
in this constructor.
"""
BaseAction.__init__(self, _names_chronics_to_backend)
if VoltageOnlyAction._shunt_added is False and type(self).shunts_data_available:
VoltageOnlyAction.attr_list_vect += ["shunt_p", "shunt_q", "shunt_bus"]
VoltageOnlyAction.authorized_keys.add("shunt")
VoltageOnlyAction._shunt_added = True
if VoltageOnlyAction._first_init is True:
self._update_value_set()
VoltageOnlyAction._first_init = False
[docs] def _check_dict(self):
"""
.. warning:: /!\\\\ Internal, do not use unless you know what you are doing /!\\\\
Check that nothing, beside prod_v has been updated with this action.
Returns
-------
"""
if self._dict_inj:
for el in self._dict_inj:
if el not in type(self).attr_list_vect:
raise AmbiguousAction(
'Impossible to modify something different than "prod_v" using '
'"VoltageOnlyAction" action.'
)
[docs] def update(self, dict_):
"""
.. warning:: /!\\\\ Internal, do not use unless you know what you are doing /!\\\\
As its original implementation, this method allows modifying the way a dictionary can be mapped to a valid
:class:`BaseAction`.
It has only minor modifications compared to the original :func:`BaseAction.update` implementation, most notably, it
doesn't update the :attr:`BaseAction._dict_inj`. It raises a warning if attempting to change them.
Parameters
----------
dict_: :class:`dict`
See the help of :func:`BaseAction.update` for a detailed explanation. **NB** all the explanations concerning the
"injection", "change bus", "set bus", or "change line status" are irrelevant for this subclass.
Returns
-------
self: :class:`PowerLineSet`
Return object itself thus allowing multiple calls to "update" to be chained.
"""
self._reset_vect()
if dict_ is not None:
for kk in dict_.keys():
if kk not in self.authorized_keys:
warn = 'The key "{}" used to update an action will be ignored. Valid keys are {}'
warn = warn.format(kk, self.authorized_keys)
warnings.warn(warn)
self._digest_injection(dict_)
self._digest_shunt(dict_)
self._check_dict()
return self