From ff9ec5d2e7e482852c74275db1711c522d86b8be Mon Sep 17 00:00:00 2001 From: jinningwang Date: Fri, 15 Nov 2024 06:43:55 -0500 Subject: [PATCH 01/96] [WIP] PFlow, write formulation --- ams/routines/pflow2.py | 5 +++++ 1 file changed, 5 insertions(+) diff --git a/ams/routines/pflow2.py b/ams/routines/pflow2.py index 8918aad4..f956851b 100644 --- a/ams/routines/pflow2.py +++ b/ams/routines/pflow2.py @@ -19,6 +19,8 @@ class PFlow2(RoutineBase): """ Power flow routine. + [In progress] + References ---------- 1. R. D. Zimmerman, C. E. Murillo-Sanchez, and R. J. Thomas, “MATPOWER: Steady-State Operations, Planning, and @@ -206,6 +208,9 @@ def __init__(self, system, config): name='sbus', is_eq=True, e_str='csb@aBus',) # --- power balance --- + # TODO: pout = VYV + # TODO: pin = $injected power$ + # TODO: pb = pout - pin pb = 'Bbus@aBus + Pbusinj + Cl@(mul(upq, pd)) + Csh@gsh - Cg@pg' self.pb = Constraint(name='pb', info='power balance', e_str=pb, is_eq=True,) From fc073da0906e2783d5b3d14d9ee4af28bc7344f2 Mon Sep 17 00:00:00 2001 From: jinningwang Date: Fri, 15 Nov 2024 06:56:00 -0500 Subject: [PATCH 02/96] Add a test of profile run --- tests/test_cli.py | 10 ++++++++++ 1 file changed, 10 insertions(+) diff --git a/tests/test_cli.py b/tests/test_cli.py index 5c164d04..9ef1cd14 100644 --- a/tests/test_cli.py +++ b/tests/test_cli.py @@ -1,4 +1,5 @@ import unittest +import os import ams.main @@ -14,3 +15,12 @@ def test_versioninfo(self): def test_misc(self): ams.main.misc(show_license=True) ams.main.misc(save_config=None, overwrite=True) + + def test_profile_run(self): + _ = ams.main.run(ams.get_case('matpower/case5.m'), + no_output=False, + profile=True,) + self.assertTrue(os.path.exists('case5_prof.prof')) + self.assertTrue(os.path.exists('case5_prof.txt')) + os.remove('case5_prof.prof') + os.remove('case5_prof.txt') From 390d530853db295d03eccef45a9117a46bd1e3c7 Mon Sep 17 00:00:00 2001 From: jinningwang Date: Fri, 15 Nov 2024 07:31:18 -0500 Subject: [PATCH 03/96] Add more tests on OModel __repr__ --- tests/test_omodel.py | 55 ++++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 55 insertions(+) diff --git a/tests/test_omodel.py b/tests/test_omodel.py index 48d24cad..97929d1c 100644 --- a/tests/test_omodel.py +++ b/tests/test_omodel.py @@ -62,3 +62,58 @@ def test_uninitialized_after_nonparametric_update(self): self.assertTrue(self.ss.DCOPF.om.initialized, "OModel should be initialized after initialization!") self.ss.DCOPF.update('ug') self.assertTrue(self.ss.DCOPF.om.initialized, "OModel should be initialized after nonparametric update!") + + +class TestOModelrepr(unittest.TestCase): + """ + Test __repr__ in module `omodel`. + """ + + def setUp(self) -> None: + self.ss = ams.load(ams.get_case("5bus/pjm5bus_demo.xlsx"), + setup=True, + default_config=True, + no_output=True, + ) + + def test_dcopf(self): + """ + Test `DCOPF.om` __repr__(). + """ + self.ss.DCOPF.om.parse() + self.assertIsInstance(self.ss.DCOPF.pg.__repr__(), str) + self.assertIsInstance(self.ss.DCOPF.pmax.__repr__(), str) + self.assertIsInstance(self.ss.DCOPF.pb.__repr__(), str) + self.assertIsInstance(self.ss.DCOPF.plflb.__repr__(), str) + self.assertIsInstance(self.ss.DCOPF.obj.__repr__(), str) + + def test_rted(self): + """ + Test `RTED.om` __repr__(). + """ + self.ss.RTED.om.parse() + self.assertIsInstance(self.ss.RTED.pru.__repr__(), str) + self.assertIsInstance(self.ss.RTED.R10.__repr__(), str) + self.assertIsInstance(self.ss.RTED.pru.__repr__(), str) + self.assertIsInstance(self.ss.RTED.rru.__repr__(), str) + self.assertIsInstance(self.ss.RTED.obj.__repr__(), str) + + def test_ed(self): + """ + Test `ED.om` __repr__(). + """ + self.ss.ED.om.parse() + self.assertIsInstance(self.ss.ED.prs.__repr__(), str) + self.assertIsInstance(self.ss.ED.R30.__repr__(), str) + self.assertIsInstance(self.ss.ED.prsb.__repr__(), str) + self.assertIsInstance(self.ss.ED.obj.__repr__(), str) + + def test_uc(self): + """ + Test `UC.om` __repr__(). + """ + self.ss.UC.om.parse() + self.assertIsInstance(self.ss.UC.prns.__repr__(), str) + self.assertIsInstance(self.ss.UC.cdp.__repr__(), str) + self.assertIsInstance(self.ss.UC.prnsb.__repr__(), str) + self.assertIsInstance(self.ss.UC.obj.__repr__(), str) From 291b816635198bc669064ab0bb271c95a0a0948b Mon Sep 17 00:00:00 2001 From: jinningwang Date: Fri, 15 Nov 2024 07:40:58 -0500 Subject: [PATCH 04/96] Add more tests on cli --- tests/test_cli.py | 8 ++++++++ 1 file changed, 8 insertions(+) diff --git a/tests/test_cli.py b/tests/test_cli.py index 9ef1cd14..ad57826b 100644 --- a/tests/test_cli.py +++ b/tests/test_cli.py @@ -2,9 +2,17 @@ import os import ams.main +import ams.cli class TestCLI(unittest.TestCase): + + def test_cli_parser(self): + ams.cli.create_parser() + + def test_cli_preamble(self): + ams.cli.preamble() + def test_main_doc(self): ams.main.doc('Bus') ams.main.doc(list_supported=True) From e5ddf1d5e5b30fd392958a4c1efc2b5dc578f31e Mon Sep 17 00:00:00 2001 From: jinningwang Date: Fri, 15 Nov 2024 07:42:30 -0500 Subject: [PATCH 05/96] [WIP] PFlow, fix its type --- ams/routines/pflow2.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/ams/routines/pflow2.py b/ams/routines/pflow2.py index f956851b..6b3ac4c9 100644 --- a/ams/routines/pflow2.py +++ b/ams/routines/pflow2.py @@ -31,7 +31,7 @@ class PFlow2(RoutineBase): def __init__(self, system, config): RoutineBase.__init__(self, system, config) self.info = 'AC Power Flow' - self.type = 'ACED' + self.type = 'PF' # --- Mapping Section --- # TODO: skip for now From 5957a6809351418192220a80e16a7f227b073b3b Mon Sep 17 00:00:00 2001 From: jinningwang Date: Fri, 15 Nov 2024 07:49:26 -0500 Subject: [PATCH 06/96] Update project README to clarify we use CVXPY to handle optimization problems --- README.md | 5 ++++- 1 file changed, 4 insertions(+), 1 deletion(-) diff --git a/README.md b/README.md index bb31ccd3..bfafabf7 100644 --- a/README.md +++ b/README.md @@ -28,7 +28,10 @@ Python Software for Power System Scheduling Modeling and Co-Simulation with Dyna # Why AMS -With the built-in interface with dynamic simulation engine, ANDES, AMS enables Dynamics Interfaced Stability Constrained Production Cost and Market Operation Modeling. +With the built-in interface with ANDES, AMS enables **Dynamics Incorporated Stability-Constrained Scheduling**. + +AMS is a **Modeling Framework** that provides a descriptive way to formulate scheduling problems. +The optimization problems are then handled by **CVXPY** and solved with third-party solvers. AMS produces credible scheduling results and competitive performance. The following results show the comparison of DCOPF between AMS and other tools. From f87e66fb0612f30c508ef9dea8c131a357f56ee1 Mon Sep 17 00:00:00 2001 From: jinningwang Date: Fri, 15 Nov 2024 08:23:03 -0500 Subject: [PATCH 07/96] Clarify that MatProcessor has taken devices connectivity into account --- ams/core/matprocessor.py | 4 +++- 1 file changed, 3 insertions(+), 1 deletion(-) diff --git a/ams/core/matprocessor.py b/ams/core/matprocessor.py index 151dac3d..0beb2243 100644 --- a/ams/core/matprocessor.py +++ b/ams/core/matprocessor.py @@ -140,7 +140,9 @@ def class_name(self): class MatProcessor: """ - Class for matrix processing in AMS system. + Class for matrices processing in AMS system. + The connectivity matrices `Cft`, `Cg`, `Cl`, and `Csh` ***have taken*** the + devices connectivity into account. The MParams' row names and col names are assigned in `System.setup()`. """ From 512a01488f8e782b5562349a90e9cd362b919657 Mon Sep 17 00:00:00 2001 From: jinningwang Date: Fri, 15 Nov 2024 08:52:54 -0500 Subject: [PATCH 08/96] Remove RParam uqp from DCOPF --- ams/routines/dcopf.py | 6 +----- 1 file changed, 1 insertion(+), 5 deletions(-) diff --git a/ams/routines/dcopf.py b/ams/routines/dcopf.py index b11a46d0..e7420051 100644 --- a/ams/routines/dcopf.py +++ b/ams/routines/dcopf.py @@ -100,10 +100,6 @@ def __init__(self, system, config): model='Slack', src='bus', no_parse=True,) # --- load --- - self.upq = RParam(info='Load connection status', - name='upq', tex_name=r'u_{PQ}', - model='StaticLoad', src='u', - no_parse=True,) self.pd = RParam(info='active demand', name='pd', tex_name=r'p_{d}', model='StaticLoad', src='p0', @@ -195,7 +191,7 @@ def __init__(self, system, config): unit='$/p.u.', model='Bus',) # --- power balance --- - pb = 'Bbus@aBus + Pbusinj + Cl@(mul(upq, pd)) + Csh@gsh - Cg@pg' + pb = 'Bbus@aBus + Pbusinj + Cl@pd + Csh@gsh - Cg@pg' self.pb = Constraint(name='pb', info='power balance', e_str=pb, is_eq=True,) # --- line flow --- From a39385731ea3288d9ce9a76b635733880871a2e6 Mon Sep 17 00:00:00 2001 From: jinningwang Date: Fri, 15 Nov 2024 09:02:42 -0500 Subject: [PATCH 09/96] Clarify ExpressionCalc --- ams/opt/omodel.py | 6 +++++- 1 file changed, 5 insertions(+), 1 deletion(-) diff --git a/ams/opt/omodel.py b/ams/opt/omodel.py index d81f9143..95cb954b 100644 --- a/ams/opt/omodel.py +++ b/ams/opt/omodel.py @@ -173,7 +173,11 @@ def size(self): class ExpressionCalc(OptzBase): """ - Expression for calculation. + Class for calculating expressions. + + Note that `ExpressionCalc` is not a CVXPY expression. It is used to calculate + expression values **after** successful optimization. + Therefore, it should not be involved in any optimization modeling. """ def __init__(self, From d5be8ce2261a351aeda249c655174dba7c7b298e Mon Sep 17 00:00:00 2001 From: jinningwang Date: Fri, 15 Nov 2024 09:03:56 -0500 Subject: [PATCH 10/96] [WIP] PFlow, formulation --- ams/routines/pflow2.py | 14 ++++++++------ 1 file changed, 8 insertions(+), 6 deletions(-) diff --git a/ams/routines/pflow2.py b/ams/routines/pflow2.py index 6b3ac4c9..c7e2cd00 100644 --- a/ams/routines/pflow2.py +++ b/ams/routines/pflow2.py @@ -98,10 +98,6 @@ def __init__(self, system, config): model='Slack', src='bus', no_parse=True,) # --- load --- - self.upq = RParam(info='Load connection status', - name='upq', tex_name=r'u_{PQ}', - model='StaticLoad', src='u', - no_parse=True,) self.pd = RParam(info='active demand', name='pd', tex_name=r'p_{d}', model='StaticLoad', src='p0', @@ -200,6 +196,11 @@ def __init__(self, system, config): unit='p.u.', name='vBus', tex_name=r'v_{bus}', model='Bus', src='v',) + self.Vc = Var(info='Bus voltage in complex', + unit='p.u.', + name='V', tex_name=r'V', + model='Bus', + cplx=True,) self.csb = VarSelect(info='select slack bus', name='csb', tex_name=r'c_{sb}', u=self.aBus, indexer='buss', @@ -208,10 +209,11 @@ def __init__(self, system, config): name='sbus', is_eq=True, e_str='csb@aBus',) # --- power balance --- - # TODO: pout = VYV + # Vc = vBus * exp(1j * aBus) + # TODO: pout = VYV; diag(Vc) @ conj(Y) @ Vc # TODO: pin = $injected power$ # TODO: pb = pout - pin - pb = 'Bbus@aBus + Pbusinj + Cl@(mul(upq, pd)) + Csh@gsh - Cg@pg' + pb = 'Bbus@aBus + Pbusinj + Cl@pd + Csh@gsh - Cg@pg' self.pb = Constraint(name='pb', info='power balance', e_str=pb, is_eq=True,) # --- line flow --- From 4885eedb1498c90e14ae246cdbcaffc784b850b1 Mon Sep 17 00:00:00 2001 From: jinningwang Date: Fri, 15 Nov 2024 11:40:13 -0500 Subject: [PATCH 11/96] Add a new omodel atom type Expression, which is different from ExpressionCalc --- ams/core/symprocessor.py | 4 + ams/opt/omodel.py | 177 +++++++++++++++++++++++++++++++++++++-- ams/routines/dcopf.py | 10 +-- ams/routines/routine.py | 12 ++- 4 files changed, 186 insertions(+), 17 deletions(-) diff --git a/ams/core/symprocessor.py b/ams/core/symprocessor.py index c1bc280e..2a7f9369 100644 --- a/ams/core/symprocessor.py +++ b/ams/core/symprocessor.py @@ -163,6 +163,10 @@ def generate_symbols(self, force_generate=False): if not service.no_parse: self.val_map[rf"\b{sname}\b"] = f"rtn.{sname}.v" + # Expressions + for ename, expr in self.parent.exprs.items(): + self.sub_map[rf"\b{ename}\b"] = f"self.rtn.{ename}.optz" + # Constraints # NOTE: constraints are included in sub_map for ExpressionCalc # thus, they don't have the suffix `.v` diff --git a/ams/opt/omodel.py b/ams/opt/omodel.py index 95cb954b..01a0fde9 100644 --- a/ams/opt/omodel.py +++ b/ams/opt/omodel.py @@ -223,7 +223,7 @@ def evaluate(self): local_vars = {'self': self, 'np': np, 'cp': cp} self.optz = self._evaluate_expression(self.code, local_vars=local_vars) except Exception as e: - raise Exception(f"Error in evaluating expr <{self.name}>.\n{e}") + raise Exception(f"Error in evaluating ExpressionCalc <{self.name}>.\n{e}") return True def _evaluate_expression(self, code, local_vars=None): @@ -256,6 +256,140 @@ def __repr__(self): return f'{self.__class__.__name__}: {self.name}' +class Expression(OptzBase): + """ + Base class for expressions used in a routine. + """ + + def __init__(self, + name: Optional[str] = None, + info: Optional[str] = None, + e_str: Optional[str] = None, + ): + OptzBase.__init__(self, name=name, info=info) + self.e_str = e_str + self.optz = None + self.code = None + + @ensure_symbols + def parse(self): + """ + Parse the expression. + + Returns + ------- + bool + Returns True if the parsing is successful, False otherwise. + """ + sub_map = self.om.rtn.syms.sub_map + code_expr = self.e_str + for pattern, replacement in sub_map.items(): + try: + code_expr = re.sub(pattern, replacement, code_expr) + except Exception as e: + raise Exception(f"Error in parsing expr <{self.name}>.\n{e}") + self.code = code_expr + msg = f" - Expression <{self.name}>: {self.e_str}" + logger.debug(pretty_long_message(msg, _prefix, max_length=_max_length)) + return True + + @ensure_mats_and_parsed + def evaluate(self): + """ + Evaluate the expression. + + Returns + ------- + bool + Returns True if the evaluation is successful, False otherwise. + """ + msg = f" - Expression <{self.name}>: {self.code}" + logger.debug(pretty_long_message(msg, _prefix, max_length=_max_length)) + try: + local_vars = {'self': self, 'np': np, 'cp': cp} + self.optz = self._evaluate_expression(self.code, local_vars=local_vars) + except Exception as e: + raise Exception(f"Error in evaluating Expression <{self.name}>.\n{e}") + return True + + def _evaluate_expression(self, code, local_vars=None): + """ + Helper method to evaluate the expression code. + + Parameters + ---------- + code : str + The code string representing the expression. + + Returns + ------- + cp.Expression + The evaluated cvxpy expression. + """ + return eval(code, {}, local_vars) + + @property + def v(self): + """ + Return the CVXPY expression value. + """ + if self.optz is None: + return None + else: + return self.optz.value + + @property + def shape(self): + """ + Return the shape. + """ + try: + return self.om.__dict__[self.name].shape + except KeyError: + logger.warning('Shape info is not ready before initialization.') + return None + + @property + def size(self): + """ + Return the size. + """ + if self.rtn.initialized: + return self.om.__dict__[self.name].size + else: + logger.warning(f'Routine <{self.rtn.class_name}> is not initialized yet.') + return None + + @property + def e(self): + """ + Return the calculated expression value. + """ + if self.code is None: + logger.info(f"Expression <{self.name}> is not parsed yet.") + return None + + val_map = self.om.rtn.syms.val_map + code = self.code + for pattern, replacement in val_map.items(): + try: + code = re.sub(pattern, replacement, code) + except TypeError as e: + raise TypeError(e) + + try: + logger.debug(pretty_long_message(f"Value code: {code}", + _prefix, max_length=_max_length)) + local_vars = {'self': self, 'np': np, 'cp': cp, 'val_map': val_map} + return self._evaluate_expression(code, local_vars) + except Exception as e: + logger.error(f"Error in calculating expr <{self.name}>.\n{e}") + return None + + def __repr__(self): + return f'{self.__class__.__name__}: {self.name}' + + class Param(OptzBase): """ Base class for parameters used in a routine. @@ -375,7 +509,7 @@ def evaluate(self): self.no_parse = True return True except Exception as e: - raise Exception(f"Error in evaluating param <{self.name}>.\n{e}") + raise Exception(f"Error in evaluating Param <{self.name}>.\n{e}") return True def update(self): @@ -611,7 +745,7 @@ def evaluate(self): local_vars = {'self': self, 'config': config, 'cp': cp} self.optz = eval(self.code, {}, local_vars) except Exception as e: - raise Exception(f"Error in evaluating var <{self.name}>.\n{e}") + raise Exception(f"Error in evaluating Var <{self.name}>.\n{e}") return True def __repr__(self): @@ -712,7 +846,7 @@ def evaluate(self): local_vars = {'self': self, 'cp': cp, 'sub_map': self.om.rtn.syms.val_map} self.optz = self._evaluate_expression(self.code, local_vars=local_vars) except Exception as e: - raise Exception(f"Error in evaluating constr <{self.name}>.\n{e}") + raise Exception(f"Error in evaluating Constraint <{self.name}>.\n{e}") def __repr__(self): enabled = 'OFF' if self.is_disabled else 'ON' @@ -905,7 +1039,7 @@ def evaluate(self): local_vars = {'self': self, 'cp': cp} self.optz = self._evaluate_expression(self.code, local_vars=local_vars) except Exception as e: - raise Exception(f"Error in evaluating obj <{self.name}>.\n{e}") + raise Exception(f"Error in evaluating Objective <{self.name}>.\n{e}") return True def _evaluate_expression(self, code, local_vars=None): @@ -941,12 +1075,14 @@ class OModel: ---------- prob: cvxpy.Problem Optimization model. + exprs: OrderedDict + Expressions registry. params: OrderedDict - Parameters. + Parameters registry. vars: OrderedDict - Decision variables. + Decision variables registry. constrs: OrderedDict - Constraints. + Constraints registry. obj: Objective Objective function. initialized: bool @@ -960,6 +1096,7 @@ class OModel: def __init__(self, routine): self.rtn = routine self.prob = None + self.exprs = OrderedDict() self.params = OrderedDict() self.vars = OrderedDict() self.constrs = OrderedDict() @@ -998,9 +1135,14 @@ def parse(self, force=False): if self.parsed and not force: logger.debug("Model is already parsed.") return self.parsed + t, _ = elapsed() - # --- add RParams and Services as parameters --- logger.warning(f'Parsing OModel for <{self.rtn.class_name}>') + # --- add expressions --- + for key, val in self.rtn.exprs.items(): + val.parse() + + # --- add RParams and Services as parameters --- for key, val in self.rtn.params.items(): if not val.no_parse: val.parse() @@ -1013,6 +1155,10 @@ def parse(self, force=False): for key, val in self.rtn.constrs.items(): val.parse() + # --- add ExpressionCalcs --- + for key, val in self.rtn.exprcs.items(): + val.parse() + # --- parse objective functions --- if self.rtn.type != 'PF': if self.rtn.obj is not None: @@ -1085,6 +1231,16 @@ def _evaluate_exprs(self): Evaluate the expressions. """ for key, val in self.rtn.exprs.items(): + try: + val.evaluate() + except Exception as e: + raise Exception(f"Failed to evaluate Expression <{key}>.\n{e}") + + def _evaluate_exprcs(self): + """ + Evaluate the expressions. + """ + for key, val in self.rtn.exprcs.items(): try: val.evaluate() except Exception as e: @@ -1120,6 +1276,7 @@ def evaluate(self, force=False): self._evaluate_constrs() self._evaluate_obj() self._evaluate_exprs() + self._evaluate_exprcs() self.evaluated = True _, s = elapsed(t) @@ -1213,6 +1370,8 @@ def _register_attribute(self, key, value): self.constrs[key] = value elif isinstance(value, cp.Parameter): self.params[key] = value + elif isinstance(value, cp.Expression): + self.exprs[key] = value def __setattr__(self, name: str, value: Any): super().__setattr__(name, value) diff --git a/ams/routines/dcopf.py b/ams/routines/dcopf.py index e7420051..0333467d 100644 --- a/ams/routines/dcopf.py +++ b/ams/routines/dcopf.py @@ -285,13 +285,13 @@ def _post_solve(self): """ Post-solve calculations. """ - for expr in self.exprs.values(): + for exprc in self.exprcs.values(): try: - var = getattr(self, expr.var) - var.optz.value = expr.v - logger.debug(f'Post solve: {var} = {expr.e_str}') + var = getattr(self, exprc.var) + var.optz.value = exprc.v + logger.debug(f'Post solve: {var} = {exprc.e_str}') except AttributeError: - raise AttributeError(f'No such variable {expr.var}') + raise AttributeError(f'No such variable {exprc.var}') return True def unpack(self, **kwargs): diff --git a/ams/routines/routine.py b/ams/routines/routine.py index 42ee82f5..19770008 100644 --- a/ams/routines/routine.py +++ b/ams/routines/routine.py @@ -16,7 +16,7 @@ from ams.core.symprocessor import SymProcessor from ams.core.documenter import RDocumenter from ams.core.service import RBaseService, ValueService -from ams.opt.omodel import OModel, Param, Var, Constraint, Objective, ExpressionCalc +from ams.opt.omodel import OModel, Param, Var, Constraint, Objective, ExpressionCalc, Expression from ams.shared import pd @@ -51,6 +51,8 @@ class RoutineBase: Registry for Var objects. constrs : OrderedDict Registry for Constraint objects. + exprcs : OrderedDict + Registry for ExpressionCalc objects. exprs : OrderedDict Registry for Expression objects. obj : Optional[Objective] @@ -105,6 +107,7 @@ def __init__(self, system=None, config=None): self.params = OrderedDict() # Param registry self.vars = OrderedDict() # Var registry self.constrs = OrderedDict() # Constraint registry + self.exprcs = OrderedDict() # ExpressionCalc registry self.exprs = OrderedDict() # Expression registry self.obj = None # Objective self.initialized = False # initialization flag @@ -544,7 +547,7 @@ def _register_attribute(self, key, value): Called within ``__setattr__``, this is where the magic happens. Subclass attributes are automatically registered based on the variable type. """ - if isinstance(value, (Param, Var, Constraint, Objective, ExpressionCalc)): + if isinstance(value, (Param, Var, Constraint, Objective, ExpressionCalc, Expression)): value.om = self.om value.rtn = self if isinstance(value, Param): @@ -556,8 +559,11 @@ def _register_attribute(self, key, value): elif isinstance(value, Constraint): self.constrs[key] = value self.om.constrs[key] = None # cp.Constraint - elif isinstance(value, ExpressionCalc): + elif isinstance(value, Expression): self.exprs[key] = value + self.om.exprs[key] = None # cp.Expression + elif isinstance(value, ExpressionCalc): + self.exprcs[key] = value elif isinstance(value, RParam): self.rparams[key] = value elif isinstance(value, RBaseService): From 20b0a3e65d0943e84b50f90e68cbdc4e0cb1ed9f Mon Sep 17 00:00:00 2001 From: jinningwang Date: Fri, 15 Nov 2024 12:27:11 -0500 Subject: [PATCH 12/96] Refactor class ExpressionCalc for efficiency --- ams/opt/omodel.py | 39 ++++++++++++++++++++++++++++----- ams/routines/dcopf.py | 50 ++++++++++++++++++++++++------------------- ams/routines/ed.py | 2 +- ams/system.py | 9 +++++--- 4 files changed, 69 insertions(+), 31 deletions(-) diff --git a/ams/opt/omodel.py b/ams/opt/omodel.py index 01a0fde9..847bcc32 100644 --- a/ams/opt/omodel.py +++ b/ams/opt/omodel.py @@ -175,23 +175,26 @@ class ExpressionCalc(OptzBase): """ Class for calculating expressions. - Note that `ExpressionCalc` is not a CVXPY expression. It is used to calculate - expression values **after** successful optimization. - Therefore, it should not be involved in any optimization modeling. + Note that `ExpressionCalc` is not a CVXPY expression, and it should not be used + in the optimization model. + It is used to calculate expression values **after** successful optimization. """ def __init__(self, name: Optional[str] = None, info: Optional[str] = None, unit: Optional[str] = None, - var: Optional[str] = None, e_str: Optional[str] = None, + model: Optional[Any] = None, + src: Optional[str] = None, ): OptzBase.__init__(self, name=name, info=info, unit=unit) self.optz = None - self.var = var self.e_str = e_str self.code = None + self.model = model + self.owner = None + self.src = src @ensure_symbols def parse(self): @@ -252,6 +255,32 @@ def v(self): else: return self.optz.value + @property + def e(self): + """ + Return the calculated expression value. + """ + if self.code is None: + logger.info(f"ExpressionCalc <{self.name}> is not parsed yet.") + return None + + val_map = self.om.rtn.syms.val_map + code = self.code + for pattern, replacement in val_map.items(): + try: + code = re.sub(pattern, replacement, code) + except TypeError as e: + raise TypeError(e) + + try: + logger.debug(pretty_long_message(f"Value code: {code}", + _prefix, max_length=_max_length)) + local_vars = {'self': self, 'np': np, 'cp': cp, 'val_map': val_map} + return self._evaluate_expression(code, local_vars) + except Exception as e: + logger.error(f"Error in calculating expr <{self.name}>.\n{e}") + return None + def __repr__(self): return f'{self.__class__.__name__}: {self.name}' diff --git a/ams/routines/dcopf.py b/ams/routines/dcopf.py index 0333467d..a6ccab2f 100644 --- a/ams/routines/dcopf.py +++ b/ams/routines/dcopf.py @@ -186,19 +186,11 @@ def __init__(self, system, config): self.sba = Constraint(info='align slack bus angle', name='sbus', is_eq=True, e_str='csb@aBus',) - self.pi = Var(info='nodal price', - name='pi', tex_name=r'\pi', - unit='$/p.u.', - model='Bus',) # --- power balance --- pb = 'Bbus@aBus + Pbusinj + Cl@pd + Csh@gsh - Cg@pg' self.pb = Constraint(name='pb', info='power balance', e_str=pb, is_eq=True,) # --- line flow --- - self.plf = Var(info='Line flow', - unit='p.u.', - name='plf', tex_name=r'p_{lf}', - model='Line',) self.plflb = Constraint(info='line flow lower bound', name='plflb', is_eq=False, e_str='-Bf@aBus - Pfinj - mul(ul, rate_a)',) @@ -211,13 +203,15 @@ def __init__(self, system, config): self.alfub = Constraint(info='line angle difference upper bound', name='alfub', is_eq=False, e_str='CftT@aBus - amax',) - self.plfc = ExpressionCalc(info='plf calculation', - name='plfc', var='plf', - e_str='Bf@aBus + Pfinj') + self.plf = ExpressionCalc(info='plf calculation', + name='plf', unit='p.u.', + e_str='Bf@aBus + Pfinj', + model='Line', src=None,) # NOTE: in CVXPY, dual_variables returns a list - self.pic = ExpressionCalc(info='dual of Constraint pb', - name='pic', var='pi', - e_str='pb.dual_variables[0]') + self.pi = ExpressionCalc(info='Locational marginal price (LMP), dual of Constraint pb', + name='pi', unit='$/p.u.', + model='Bus', src=None, + e_str='pb.dual_variables[0]') # --- objective --- obj = 'sum(mul(c2, pg**2))' @@ -285,20 +279,13 @@ def _post_solve(self): """ Post-solve calculations. """ - for exprc in self.exprcs.values(): - try: - var = getattr(self, exprc.var) - var.optz.value = exprc.v - logger.debug(f'Post solve: {var} = {exprc.e_str}') - except AttributeError: - raise AttributeError(f'No such variable {exprc.var}') return True def unpack(self, **kwargs): """ Unpack the results from CVXPY model. """ - # --- solver results to routine algeb --- + # --- solver Var results to routine algeb --- for _, var in self.vars.items(): # --- copy results from routine algeb into system algeb --- if var.model is None: # if no owner @@ -319,6 +306,25 @@ def unpack(self, **kwargs): logger.error(f'Failed to unpack <{var}> to <{var.owner.class_name}>.') pass + # --- solver ExpressionCalc results to routine algeb --- + for _, exprc in self.exprcs.items(): + if exprc.model is None: + continue + if exprc.src is None: + continue + else: + try: + idx = exprc.owner.get_idx() + except AttributeError: + idx = exprc.owner.idx.v + else: + pass + try: + exprc.owner.set(src=exprc.src, idx=idx, attr='v', value=exprc.v) + except (KeyError, TypeError): + logger.error(f'Failed to unpack <{exprc}> to <{exprc.owner.class_name}>.') + pass + # label the most recent solved routine self.system.recent = self.system.routines[self.class_name] return True diff --git a/ams/routines/ed.py b/ams/routines/ed.py index 24b4b989..1ea2592f 100644 --- a/ams/routines/ed.py +++ b/ams/routines/ed.py @@ -182,7 +182,7 @@ def __init__(self, system, config): self.alflb.e_str = '-CftT@aBus + amin@tlv' self.alfub.e_str = 'CftT@aBus - amax@tlv' - self.plfc.e_str = 'Bf@aBus + Pfinj@tlv' + self.plf.e_str = 'Bf@aBus + Pfinj@tlv' # --- power balance --- self.pb.e_str = 'Bbus@aBus + Pbusinj@tlv + Cl@pds + Csh@gsh@tlv - Cg@pg' diff --git a/ams/system.py b/ams/system.py index f14603b9..834827c3 100644 --- a/ams/system.py +++ b/ams/system.py @@ -229,7 +229,7 @@ def import_types(self): def _collect_group_data(self, items): """ - Set the owner for routine attributes: ``RParam``, ``Var``, and ``RBaseService``. + Set the owner for routine attributes: ``RParam``, ``Var``, ``ExpressionCalc``, and ``RBaseService``. """ for item_name, item in items.items(): if item.model in self.groups.keys(): @@ -271,12 +271,15 @@ def import_routines(self): type_instance = self.types[type_name] type_instance.routines[vname] = rtn # self.types[rtn.type].routines[vname] = rtn - # Collect rparams + # Collect RParams rparams = getattr(rtn, 'rparams') self._collect_group_data(rparams) - # Collect routine vars + # Collect routine Vars r_vars = getattr(rtn, 'vars') self._collect_group_data(r_vars) + # Collect ExpressionCalcs + exprc = getattr(rtn, 'exprcs') + self._collect_group_data(exprc) def import_groups(self): """ From 944516e19a3511e7cb4a3d22513d7d8e24618e0f Mon Sep 17 00:00:00 2001 From: jinningwang Date: Fri, 15 Nov 2024 13:08:02 -0500 Subject: [PATCH 13/96] Fix RDocumenter for ExpressionCalc --- ams/core/documenter.py | 26 +++++++++++++++++--------- tests/test_1st_system.py | 2 +- 2 files changed, 18 insertions(+), 10 deletions(-) diff --git a/ams/core/documenter.py b/ams/core/documenter.py index 727c2d32..ecb9a175 100644 --- a/ams/core/documenter.py +++ b/ams/core/documenter.py @@ -224,7 +224,7 @@ def get(self, max_width=78, export='plain'): self.parent.syms.generate_symbols() out += self._obj_doc(max_width=max_width, export=export) out += self._constr_doc(max_width=max_width, export=export) - out += self._expr_doc(max_width=max_width, export=export) + out += self._exprc_doc(max_width=max_width, export=export) out += self._var_doc(max_width=max_width, export=export) out += self._service_doc(max_width=max_width, export=export) out += self._param_doc(max_width=max_width, export=export) @@ -315,18 +315,25 @@ def _constr_doc(self, max_width=78, export='plain'): plain_dict=plain_dict, rest_dict=rest_dict) - def _expr_doc(self, max_width=78, export='plain'): - # expression documentation - if len(self.parent.exprs) == 0: + def _exprc_doc(self, max_width=78, export='plain'): + # ExpressionCalc documentation + if len(self.parent.exprcs) == 0: return '' # prepare temporary lists - names, var_names, info = list(), list(), list() + names, info = list(), list() + units, sources, units_rest = list(), list(), list() - for p in self.parent.exprs.values(): + for p in self.parent.exprcs.values(): names.append(p.name) - var_names.append(p.var) info.append(p.info if p.info else '') + units.append(p.unit if p.unit else '') + units_rest.append(f'*{p.unit}*' if p.unit else '') + + slist = [] + if p.owner is not None and p.src is not None: + slist.append(f'{p.owner.class_name}.{p.src}') + sources.append(','.join(slist)) # expressions based on output format expressions = [] @@ -342,13 +349,14 @@ def _expr_doc(self, max_width=78, export='plain'): expressions = math_wrap(expressions, export=export) plain_dict = OrderedDict([('Name', names), - ('Variable', var_names), ('Description', info), + ('Unit', units), ]) rest_dict = OrderedDict([('Name', names), - ('Variable', var_names), ('Description', info), ('Expression', expressions), + ('Unit', units_rest), + ('Source', sources), ]) # convert to rows and export as table diff --git a/tests/test_1st_system.py b/tests/test_1st_system.py index d2cc20a1..79f5fa39 100644 --- a/tests/test_1st_system.py +++ b/tests/test_1st_system.py @@ -26,7 +26,7 @@ def test_docum(self) -> None: for export in ['plain', 'rest']: docum._obj_doc(max_width=78, export=export) docum._constr_doc(max_width=78, export=export) - docum._expr_doc(max_width=78, export=export) + docum._exprc_doc(max_width=78, export=export) docum._var_doc(max_width=78, export=export) docum._service_doc(max_width=78, export=export) docum._param_doc(max_width=78, export=export) From 3b9e23be0a5146aaf9c9c6ad6fa2e9917bc7cab5 Mon Sep 17 00:00:00 2001 From: jinningwang Date: Fri, 15 Nov 2024 13:10:16 -0500 Subject: [PATCH 14/96] Update release notes --- docs/source/release-notes.rst | 5 ++++- 1 file changed, 4 insertions(+), 1 deletion(-) diff --git a/docs/source/release-notes.rst b/docs/source/release-notes.rst index b1ddb161..8990309a 100644 --- a/docs/source/release-notes.rst +++ b/docs/source/release-notes.rst @@ -12,9 +12,12 @@ Pre-v1.0.0 v0.9.12 (202x-xx-xx) -------------------- +TODO: class `Expression`: registry in `OModel` and `Routine`, fit symbol processor, fit documenter + - Refactor OModel.initialized as a property method - Add a demo to show using `Constraint.e` for debugging -- Fix `OModel.Param.evaluate()` when its value is a number +- Fix `ams.opt.omodel.Param.evaluate()` when its value is a number +- Improve `ams.opt.omodel.ExpressionCalc()` for efficiency RC1 ~~~~ From 551b9dac353900f7e507052ddb2ff9ffe64e82a0 Mon Sep 17 00:00:00 2001 From: jinningwang Date: Fri, 15 Nov 2024 13:17:16 -0500 Subject: [PATCH 15/96] Include demo_debug in documentation --- docs/source/examples/index.rst | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/docs/source/examples/index.rst b/docs/source/examples/index.rst index fe929b9c..6dacab19 100644 --- a/docs/source/examples/index.rst +++ b/docs/source/examples/index.rst @@ -30,4 +30,5 @@ folder of the repository :caption: Demonstration ../_examples/demo/demo_ESD1.ipynb - ../_examples/demo/demo_AGC.ipynb \ No newline at end of file + ../_examples/demo/demo_AGC.ipynb + ../_examples/demo/demo_debug.ipynb \ No newline at end of file From e9fe1fbf8ab63246f88109017807701c2a756ebc Mon Sep 17 00:00:00 2001 From: jinningwang Date: Fri, 15 Nov 2024 13:27:22 -0500 Subject: [PATCH 16/96] Fix documenter for ExpressionCalc --- ams/core/documenter.py | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/ams/core/documenter.py b/ams/core/documenter.py index ecb9a175..0cf51a22 100644 --- a/ams/core/documenter.py +++ b/ams/core/documenter.py @@ -338,14 +338,14 @@ def _exprc_doc(self, max_width=78, export='plain'): # expressions based on output format expressions = [] if export == 'rest': - for p in self.parent.exprs.values(): + for p in self.parent.exprcs.values(): expr = _tex_pre(self, p, self.parent.syms.tex_map) logger.debug(f'{p.name} math: {expr}') expressions.append(expr) - title = 'Expressions\n----------------------------------' + title = 'ExpressionCalcs\n----------------------------------' else: - title = 'Expressions' + title = 'ExpressionCalcs' expressions = math_wrap(expressions, export=export) plain_dict = OrderedDict([('Name', names), From 82805f79b076072aaff5c01b1666a87dc338676d Mon Sep 17 00:00:00 2001 From: jinningwang Date: Fri, 15 Nov 2024 13:28:29 -0500 Subject: [PATCH 17/96] Typo --- ams/routines/dcopf.py | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/ams/routines/dcopf.py b/ams/routines/dcopf.py index a6ccab2f..0f9c5595 100644 --- a/ams/routines/dcopf.py +++ b/ams/routines/dcopf.py @@ -203,12 +203,12 @@ def __init__(self, system, config): self.alfub = Constraint(info='line angle difference upper bound', name='alfub', is_eq=False, e_str='CftT@aBus - amax',) - self.plf = ExpressionCalc(info='plf calculation', + self.plf = ExpressionCalc(info='Line flow', name='plf', unit='p.u.', e_str='Bf@aBus + Pfinj', model='Line', src=None,) # NOTE: in CVXPY, dual_variables returns a list - self.pi = ExpressionCalc(info='Locational marginal price (LMP), dual of Constraint pb', + self.pi = ExpressionCalc(info='LMP, dual of ', name='pi', unit='$/p.u.', model='Bus', src=None, e_str='pb.dual_variables[0]') From 4857875293b38c3dc33b584a1a206b72da629db2 Mon Sep 17 00:00:00 2001 From: jinningwang Date: Fri, 15 Nov 2024 13:38:36 -0500 Subject: [PATCH 18/96] Format --- ams/opt/omodel.py | 29 ++++++++++++----------------- 1 file changed, 12 insertions(+), 17 deletions(-) diff --git a/ams/opt/omodel.py b/ams/opt/omodel.py index 847bcc32..04c88323 100644 --- a/ams/opt/omodel.py +++ b/ams/opt/omodel.py @@ -32,8 +32,6 @@ def ensure_symbols(func): and optimization model (`OModel`), i.e., `Var`, `Param`, `Constraint`, `Objective`, and `ExpressionCalc`. - Note: - ----- Parsing before symbol generation can give wrong results. Ensure that symbols are generated before calling the `parse` method. """ @@ -48,17 +46,15 @@ def wrapper(self, *args, **kwargs): def ensure_mats_and_parsed(func): """ - Decorator to ensure that system matrices are built and OModel is parsed + Decorator to ensure that system matrices are built and the OModel is parsed before evaluation. If not, it runs the necessary methods to initialize them. - Designed to be used on the `evaluate` method of the optimization elements (`OptzBase`) - and optimization model (`OModel`), i.e., `Var`, `Param`, `Constraint`, `Objective`, + Designed to be used on the `evaluate` method of optimization elements (`OptzBase`) + and the optimization model (`OModel`), i.e., `Var`, `Param`, `Constraint`, `Objective`, and `ExpressionCalc`. - Note: - ----- - Evaluation before matrices building and parsing can run into errors. Ensure that - system matrices are built and OModel is parsed before calling the `evaluate` method. + Evaluation before building matrices and parsing the OModel can lead to errors. Ensure that + system matrices are built and the OModel is parsed before calling the `evaluate` method. """ def wrapper(self, *args, **kwargs): @@ -83,24 +79,23 @@ def wrapper(self, *args, **kwargs): class OptzBase: """ - Base class for optimization elements, e.g., Var and Constraint. + Base class for optimization elements. + Ensure that symbols are generated before calling the `parse` method. Parsing + before symbol generation can lead to incorrect results. Parameters ---------- name : str, optional - Name. + Name of the optimization element. info : str, optional - Descriptive information + Descriptive information about the optimization element. + unit : str, optional + Unit of measurement for the optimization element. Attributes ---------- rtn : ams.routines.Routine The owner routine instance. - - Note: - ----- - Ensure that symbols are generated before calling the `parse` method. Parsing - before symbol generation can give wrong results. """ def __init__(self, From 8f1cecfb0b232a4babc7b8b91b52a5878c5ca00a Mon Sep 17 00:00:00 2001 From: jinningwang Date: Fri, 15 Nov 2024 13:40:57 -0500 Subject: [PATCH 19/96] Format --- ams/opt/omodel.py | 2 -- 1 file changed, 2 deletions(-) diff --git a/ams/opt/omodel.py b/ams/opt/omodel.py index 04c88323..96d341ec 100644 --- a/ams/opt/omodel.py +++ b/ams/opt/omodel.py @@ -642,8 +642,6 @@ def __init__(self, self.unit = unit self.tex_name = tex_name if tex_name else name - # instance of the owner Model - self.owner = None # variable internal index inside a model (assigned in run time) self.id = None OptzBase.__init__(self, name=name, info=info, unit=unit) From 4c403c097013cd3cf55adbf50680ef9eb548bad0 Mon Sep 17 00:00:00 2001 From: jinningwang Date: Fri, 15 Nov 2024 13:45:01 -0500 Subject: [PATCH 20/96] Add method for easier usage --- ams/opt/omodel.py | 10 ++++++++++ 1 file changed, 10 insertions(+) diff --git a/ams/opt/omodel.py b/ams/opt/omodel.py index 96d341ec..84411c21 100644 --- a/ams/opt/omodel.py +++ b/ams/opt/omodel.py @@ -190,6 +190,16 @@ def __init__(self, self.model = model self.owner = None self.src = src + self.is_group = False + + def get_idx(self): + if self.is_group: + return self.owner.get_idx() + elif self.owner is None: + logger.info(f'ExpressionCalc <{self.name}> has no owner.') + return None + else: + return self.owner.idx.v @ensure_symbols def parse(self): From 04e6f410b3035776b2c62eb8fb2b8ebf0cc1e4ff Mon Sep 17 00:00:00 2001 From: jinningwang Date: Fri, 15 Nov 2024 13:50:46 -0500 Subject: [PATCH 21/96] Move constants _prefix and _max_length from module omodel to shared --- ams/opt/omodel.py | 5 +---- ams/shared.py | 4 ++++ 2 files changed, 5 insertions(+), 4 deletions(-) diff --git a/ams/opt/omodel.py b/ams/opt/omodel.py index 84411c21..67f098d0 100644 --- a/ams/opt/omodel.py +++ b/ams/opt/omodel.py @@ -15,13 +15,10 @@ import cvxpy as cp from ams.utils import pretty_long_message -from ams.shared import sps +from ams.shared import sps, _prefix, _max_length logger = logging.getLogger(__name__) -_prefix = r" - --------------> | " -_max_length = 80 - def ensure_symbols(func): """ diff --git a/ams/shared.py b/ams/shared.py index afe3c92e..9d74670e 100644 --- a/ams/shared.py +++ b/ams/shared.py @@ -31,6 +31,10 @@ inf = np.inf nan = np.nan +# --- misc constants --- +_prefix = r" - --------------> | " # NOQA +_max_length = 80 # NOQA + # NOTE: copyright year_end = datetime.now().year copyright_msg = f'Copyright (C) 2023-{year_end} Jinning Wang' From d128277073fa5f091f13f65bf774c72173c4205f Mon Sep 17 00:00:00 2001 From: jinningwang Date: Fri, 15 Nov 2024 14:17:09 -0500 Subject: [PATCH 22/96] Refactor module opt --- ams/__init__.py | 1 + ams/core/matprocessor.py | 2 +- ams/core/param.py | 2 +- ams/core/service.py | 2 +- ams/opt/__init__.py | 9 +- ams/opt/constraint.py | 172 ++++++ ams/opt/expr.py | 152 ++++++ ams/opt/exprcalc.py | 142 +++++ ams/opt/objective.py | 174 ++++++ ams/opt/omodel.py | 1083 +------------------------------------- ams/opt/optbase.py | 152 ++++++ ams/opt/param.py | 156 ++++++ ams/opt/var.py | 245 +++++++++ ams/routines/acopf.py | 2 +- ams/routines/dcopf.py | 2 +- ams/routines/dcpf.py | 2 +- ams/routines/dopf.py | 2 +- ams/routines/ed.py | 2 +- ams/routines/pflow.py | 2 +- ams/routines/pflow2.py | 10 +- ams/routines/routine.py | 2 +- ams/routines/rted.py | 2 +- ams/routines/uc.py | 2 +- 23 files changed, 1222 insertions(+), 1098 deletions(-) create mode 100644 ams/opt/constraint.py create mode 100644 ams/opt/expr.py create mode 100644 ams/opt/exprcalc.py create mode 100644 ams/opt/objective.py create mode 100644 ams/opt/optbase.py create mode 100644 ams/opt/param.py create mode 100644 ams/opt/var.py diff --git a/ams/__init__.py b/ams/__init__.py index ae078bea..b05d4e9e 100644 --- a/ams/__init__.py +++ b/ams/__init__.py @@ -1,6 +1,7 @@ from . import _version __version__ = _version.get_versions()['version'] +from ams import opt # NOQA from ams import benchmarks # NOQA from ams.main import config_logger, load, run # NOQA diff --git a/ams/core/matprocessor.py b/ams/core/matprocessor.py index 0beb2243..bf0e4ac1 100644 --- a/ams/core/matprocessor.py +++ b/ams/core/matprocessor.py @@ -13,7 +13,7 @@ from andes.shared import tqdm, tqdm_nb from andes.utils.misc import elapsed, is_notebook -from ams.opt.omodel import Param +from ams.opt import Param from ams.shared import pd, sps logger = logging.getLogger(__name__) diff --git a/ams/core/param.py b/ams/core/param.py index 50c0cc4c..dbefba49 100644 --- a/ams/core/param.py +++ b/ams/core/param.py @@ -10,7 +10,7 @@ import numpy as np from scipy.sparse import issparse -from ams.opt.omodel import Param +from ams.opt import Param logger = logging.getLogger(__name__) diff --git a/ams/core/service.py b/ams/core/service.py index d94ce5e8..0f48653d 100644 --- a/ams/core/service.py +++ b/ams/core/service.py @@ -10,7 +10,7 @@ from andes.core.service import BaseService -from ams.opt.omodel import Param +from ams.opt import Param logger = logging.getLogger(__name__) diff --git a/ams/opt/__init__.py b/ams/opt/__init__.py index d442bbcb..c468e92f 100644 --- a/ams/opt/__init__.py +++ b/ams/opt/__init__.py @@ -2,4 +2,11 @@ Module for optimization modeling. """ -from ams.opt.omodel import Var, Constraint, Objective, OModel # NOQA +from ams.opt.optbase import OptzBase, ensure_symbols, ensure_mats_and_parsed # NOQA +from ams.opt.var import Var # NOQA +from ams.opt.exprcalc import ExpressionCalc # NOQA +from ams.opt.param import Param # NOQA +from ams.opt.constraint import Constraint # NOQA +from ams.opt.objective import Objective # NOQA +from ams.opt.expr import Expression # NOQA +from ams.opt.omodel import OModel # NOQA diff --git a/ams/opt/constraint.py b/ams/opt/constraint.py new file mode 100644 index 00000000..8ab5a508 --- /dev/null +++ b/ams/opt/constraint.py @@ -0,0 +1,172 @@ +""" +Module for optimization Constraint. +""" +import logging + +from typing import Optional +import re + +import numpy as np + +import cvxpy as cp + +from ams.utils import pretty_long_message +from ams.shared import _prefix, _max_length + +from ams.opt import OptzBase, ensure_symbols, ensure_mats_and_parsed + +logger = logging.getLogger(__name__) + + +class Constraint(OptzBase): + """ + Base class for constraints. + + This class is used as a template for defining constraints. Each + instance of this class represents a single constraint. + + Parameters + ---------- + name : str, optional + A user-defined name for the constraint. + e_str : str, optional + A mathematical expression representing the constraint. + info : str, optional + Additional informational text about the constraint. + is_eq : str, optional + Flag indicating if the constraint is an equality constraint. False indicates + an inequality constraint in the form of `<= 0`. + + Attributes + ---------- + is_disabled : bool + Flag indicating if the constraint is disabled, False by default. + rtn : ams.routines.Routine + The owner routine instance. + is_disabled : bool, optional + Flag indicating if the constraint is disabled, False by default. + dual : float, optional + The dual value of the constraint. + code : str, optional + The code string for the constraint + """ + + def __init__(self, + name: Optional[str] = None, + e_str: Optional[str] = None, + info: Optional[str] = None, + is_eq: Optional[bool] = False, + ): + OptzBase.__init__(self, name=name, info=info) + self.e_str = e_str + self.is_eq = is_eq + self.is_disabled = False + self.dual = None + self.code = None + + @ensure_symbols + def parse(self): + """ + Parse the constraint. + """ + # parse the expression str + sub_map = self.om.rtn.syms.sub_map + code_constr = self.e_str + for pattern, replacement in sub_map.items(): + try: + code_constr = re.sub(pattern, replacement, code_constr) + except TypeError as e: + raise TypeError(f"Error in parsing constr <{self.name}>.\n{e}") + # parse the constraint type + code_constr += " == 0" if self.is_eq else " <= 0" + # store the parsed expression str code + self.code = code_constr + msg = f" - Constr <{self.name}>: {self.e_str}" + logger.debug(pretty_long_message(msg, _prefix, max_length=_max_length)) + return True + + def _evaluate_expression(self, code, local_vars=None): + """ + Helper method to evaluate the expression code. + + Parameters + ---------- + code : str + The code string representing the expression. + + Returns + ------- + cp.Expression + The evaluated cvxpy expression. + """ + return eval(code, {}, local_vars) + + @ensure_mats_and_parsed + def evaluate(self): + """ + Evaluate the constraint. + """ + msg = f" - Constr <{self.name}>: {self.code}" + logger.debug(pretty_long_message(msg, _prefix, max_length=_max_length)) + try: + local_vars = {'self': self, 'cp': cp, 'sub_map': self.om.rtn.syms.val_map} + self.optz = self._evaluate_expression(self.code, local_vars=local_vars) + except Exception as e: + raise Exception(f"Error in evaluating Constraint <{self.name}>.\n{e}") + + def __repr__(self): + enabled = 'OFF' if self.is_disabled else 'ON' + out = f"{self.class_name}: {self.name} [{enabled}]" + return out + + @property + def e(self): + """ + Return the calculated constraint LHS value. + Note that `v` should be used primarily as it is obtained + from the solver directly. + + `e` is for debugging purpose. For a successfully solved problem, + `e` should equal to `v`. However, when a problem is infeasible + or unbounded, `e` can be used to check the constraint LHS value. + """ + if self.code is None: + logger.info(f"Constraint <{self.name}> is not parsed yet.") + return None + + val_map = self.om.rtn.syms.val_map + code = self.code + for pattern, replacement in val_map.items(): + try: + code = re.sub(pattern, replacement, code) + except TypeError as e: + raise TypeError(e) + + try: + logger.debug(pretty_long_message(f"Value code: {code}", + _prefix, max_length=_max_length)) + local_vars = {'self': self, 'np': np, 'cp': cp, 'val_map': val_map} + return self._evaluate_expression(code, local_vars) + except Exception as e: + logger.error(f"Error in calculating constr <{self.name}>.\n{e}") + return None + + @property + def v(self): + """ + Return the CVXPY constraint LHS value. + """ + if self.optz is None: + return None + if self.optz._expr.value is None: + try: + shape = self._expr.shape + return np.zeros(shape) + except AttributeError: + return None + else: + return self.optz._expr.value + + @v.setter + def v(self, value): + raise AttributeError("Cannot set the value of the constraint.") diff --git a/ams/opt/expr.py b/ams/opt/expr.py new file mode 100644 index 00000000..72a715a1 --- /dev/null +++ b/ams/opt/expr.py @@ -0,0 +1,152 @@ +""" +Module for optimization Expression. +""" +import logging + +from typing import Optional +import re + +import numpy as np + +import cvxpy as cp + +from ams.utils import pretty_long_message +from ams.shared import _prefix, _max_length + +from ams.opt import OptzBase, ensure_symbols, ensure_mats_and_parsed + +logger = logging.getLogger(__name__) + + +class Expression(OptzBase): + """ + Base class for expressions used in a routine. + """ + + def __init__(self, + name: Optional[str] = None, + info: Optional[str] = None, + e_str: Optional[str] = None, + ): + OptzBase.__init__(self, name=name, info=info) + self.e_str = e_str + self.optz = None + self.code = None + + @ensure_symbols + def parse(self): + """ + Parse the expression. + + Returns + ------- + bool + Returns True if the parsing is successful, False otherwise. + """ + sub_map = self.om.rtn.syms.sub_map + code_expr = self.e_str + for pattern, replacement in sub_map.items(): + try: + code_expr = re.sub(pattern, replacement, code_expr) + except Exception as e: + raise Exception(f"Error in parsing expr <{self.name}>.\n{e}") + self.code = code_expr + msg = f" - Expression <{self.name}>: {self.e_str}" + logger.debug(pretty_long_message(msg, _prefix, max_length=_max_length)) + return True + + @ensure_mats_and_parsed + def evaluate(self): + """ + Evaluate the expression. + + Returns + ------- + bool + Returns True if the evaluation is successful, False otherwise. + """ + msg = f" - Expression <{self.name}>: {self.code}" + logger.debug(pretty_long_message(msg, _prefix, max_length=_max_length)) + try: + local_vars = {'self': self, 'np': np, 'cp': cp} + self.optz = self._evaluate_expression(self.code, local_vars=local_vars) + except Exception as e: + raise Exception(f"Error in evaluating Expression <{self.name}>.\n{e}") + return True + + def _evaluate_expression(self, code, local_vars=None): + """ + Helper method to evaluate the expression code. + + Parameters + ---------- + code : str + The code string representing the expression. + + Returns + ------- + cp.Expression + The evaluated cvxpy expression. + """ + return eval(code, {}, local_vars) + + @property + def v(self): + """ + Return the CVXPY expression value. + """ + if self.optz is None: + return None + else: + return self.optz.value + + @property + def shape(self): + """ + Return the shape. + """ + try: + return self.om.__dict__[self.name].shape + except KeyError: + logger.warning('Shape info is not ready before initialization.') + return None + + @property + def size(self): + """ + Return the size. + """ + if self.rtn.initialized: + return self.om.__dict__[self.name].size + else: + logger.warning(f'Routine <{self.rtn.class_name}> is not initialized yet.') + return None + + @property + def e(self): + """ + Return the calculated expression value. + """ + if self.code is None: + logger.info(f"Expression <{self.name}> is not parsed yet.") + return None + + val_map = self.om.rtn.syms.val_map + code = self.code + for pattern, replacement in val_map.items(): + try: + code = re.sub(pattern, replacement, code) + except TypeError as e: + raise TypeError(e) + + try: + logger.debug(pretty_long_message(f"Value code: {code}", + _prefix, max_length=_max_length)) + local_vars = {'self': self, 'np': np, 'cp': cp, 'val_map': val_map} + return self._evaluate_expression(code, local_vars) + except Exception as e: + logger.error(f"Error in calculating expr <{self.name}>.\n{e}") + return None + + def __repr__(self): + return f'{self.__class__.__name__}: {self.name}' diff --git a/ams/opt/exprcalc.py b/ams/opt/exprcalc.py new file mode 100644 index 00000000..25e11230 --- /dev/null +++ b/ams/opt/exprcalc.py @@ -0,0 +1,142 @@ +""" +Module for optimization ExpressionCalc. +""" +import logging + +from typing import Any, Optional +import re + +import numpy as np + +import cvxpy as cp + +from ams.utils import pretty_long_message +from ams.shared import _prefix, _max_length + +from ams.opt import OptzBase, ensure_symbols, ensure_mats_and_parsed + +logger = logging.getLogger(__name__) + + +class ExpressionCalc(OptzBase): + """ + Class for calculating expressions. + + Note that `ExpressionCalc` is not a CVXPY expression, and it should not be used + in the optimization model. + It is used to calculate expression values **after** successful optimization. + """ + + def __init__(self, + name: Optional[str] = None, + info: Optional[str] = None, + unit: Optional[str] = None, + e_str: Optional[str] = None, + model: Optional[Any] = None, + src: Optional[str] = None, + ): + OptzBase.__init__(self, name=name, info=info, unit=unit) + self.optz = None + self.e_str = e_str + self.code = None + self.model = model + self.owner = None + self.src = src + self.is_group = False + + def get_idx(self): + if self.is_group: + return self.owner.get_idx() + elif self.owner is None: + logger.info(f'ExpressionCalc <{self.name}> has no owner.') + return None + else: + return self.owner.idx.v + + @ensure_symbols + def parse(self): + """ + Parse the Expression. + """ + # parse the expression str + sub_map = self.om.rtn.syms.sub_map + code_expr = self.e_str + for pattern, replacement in sub_map.items(): + try: + code_expr = re.sub(pattern, replacement, code_expr) + except Exception as e: + raise Exception(f"Error in parsing expr <{self.name}>.\n{e}") + # store the parsed expression str code + self.code = code_expr + msg = f" - ExpressionCalc <{self.name}>: {self.e_str}" + logger.debug(pretty_long_message(msg, _prefix, max_length=_max_length)) + return True + + @ensure_mats_and_parsed + def evaluate(self): + """ + Evaluate the expression. + """ + msg = f" - Expression <{self.name}>: {self.code}" + logger.debug(pretty_long_message(msg, _prefix, max_length=_max_length)) + try: + local_vars = {'self': self, 'np': np, 'cp': cp} + self.optz = self._evaluate_expression(self.code, local_vars=local_vars) + except Exception as e: + raise Exception(f"Error in evaluating ExpressionCalc <{self.name}>.\n{e}") + return True + + def _evaluate_expression(self, code, local_vars=None): + """ + Helper method to evaluate the expression code. + + Parameters + ---------- + code : str + The code string representing the expression. + + Returns + ------- + cp.Expression + The evaluated cvxpy expression. + """ + return eval(code, {}, local_vars) + + @property + def v(self): + """ + Return the CVXPY expression value. + """ + if self.optz is None: + return None + else: + return self.optz.value + + @property + def e(self): + """ + Return the calculated expression value. + """ + if self.code is None: + logger.info(f"ExpressionCalc <{self.name}> is not parsed yet.") + return None + + val_map = self.om.rtn.syms.val_map + code = self.code + for pattern, replacement in val_map.items(): + try: + code = re.sub(pattern, replacement, code) + except TypeError as e: + raise TypeError(e) + + try: + logger.debug(pretty_long_message(f"Value code: {code}", + _prefix, max_length=_max_length)) + local_vars = {'self': self, 'np': np, 'cp': cp, 'val_map': val_map} + return self._evaluate_expression(code, local_vars) + except Exception as e: + logger.error(f"Error in calculating expr <{self.name}>.\n{e}") + return None + + def __repr__(self): + return f'{self.__class__.__name__}: {self.name}' diff --git a/ams/opt/objective.py b/ams/opt/objective.py new file mode 100644 index 00000000..f9695ec4 --- /dev/null +++ b/ams/opt/objective.py @@ -0,0 +1,174 @@ +""" +Module for optimization Objective. +""" +import logging + +from typing import Optional +import re + +import numpy as np + +import cvxpy as cp + +from ams.utils import pretty_long_message +from ams.shared import _prefix, _max_length + +from ams.opt import OptzBase, ensure_symbols, ensure_mats_and_parsed + +logger = logging.getLogger(__name__) + + +class Objective(OptzBase): + """ + Base class for objective functions. + + This class serves as a template for defining objective functions. Each + instance of this class represents a single objective function that can + be minimized or maximized depending on the sense ('min' or 'max'). + + Parameters + ---------- + name : str, optional + A user-defined name for the objective function. + e_str : str, optional + A mathematical expression representing the objective function. + info : str, optional + Additional informational text about the objective function. + sense : str, optional + The sense of the objective function, default to 'min'. + `min` for minimization and `max` for maximization. + + Attributes + ---------- + v : NoneType + The value of the objective function. It needs to be set through + computation. + rtn : ams.routines.Routine + The owner routine instance. + code : str + The code string for the objective function. + """ + + def __init__(self, + name: Optional[str] = None, + e_str: Optional[str] = None, + info: Optional[str] = None, + unit: Optional[str] = None, + sense: Optional[str] = 'min'): + OptzBase.__init__(self, name=name, info=info, unit=unit) + self.e_str = e_str + self.sense = sense + self.code = None + + @property + def e(self): + """ + Return the calculated objective value. + + Note that `v` should be used primarily as it is obtained + from the solver directly. + + `e` is for debugging purpose. For a successfully solved problem, + `e` should equal to `v`. However, when a problem is infeasible + or unbounded, `e` can be used to check the objective value. + """ + if self.code is None: + logger.info(f"Objective <{self.name}> is not parsed yet.") + return None + + val_map = self.om.rtn.syms.val_map + code = self.code + for pattern, replacement in val_map.items(): + try: + code = re.sub(pattern, replacement, code) + except TypeError as e: + logger.error(f"Error in parsing value for obj <{self.name}>.") + raise e + + try: + logger.debug(pretty_long_message(f"Value code: {code}", + _prefix, max_length=_max_length)) + local_vars = {'self': self, 'np': np, 'cp': cp, 'val_map': val_map} + return self._evaluate_expression(code, local_vars) + except Exception as e: + logger.error(f"Error in calculating obj <{self.name}>.\n{e}") + return None + + @property + def v(self): + """ + Return the CVXPY objective value. + """ + if self.optz is None: + return None + else: + return self.optz.value + + @v.setter + def v(self, value): + raise AttributeError("Cannot set the value of the objective function.") + + @ensure_symbols + def parse(self): + """ + Parse the objective function. + + Returns + ------- + bool + Returns True if the parsing is successful, False otherwise. + """ + # parse the expression str + sub_map = self.om.rtn.syms.sub_map + code_obj = self.e_str + for pattern, replacement, in sub_map.items(): + try: + code_obj = re.sub(pattern, replacement, code_obj) + except Exception as e: + raise Exception(f"Error in parsing obj <{self.name}>.\n{e}") + # store the parsed expression str code + self.code = code_obj + if self.sense not in ['min', 'max']: + raise ValueError(f'Objective sense {self.sense} is not supported.') + sense = 'cp.Minimize' if self.sense == 'min' else 'cp.Maximize' + self.code = f"{sense}({code_obj})" + msg = f" - Objective <{self.name}>: {self.code}" + logger.debug(pretty_long_message(msg, _prefix, max_length=_max_length)) + return True + + @ensure_mats_and_parsed + def evaluate(self): + """ + Evaluate the objective function. + + Returns + ------- + bool + Returns True if the evaluation is successful, False otherwise. + """ + logger.debug(f" - Objective <{self.name}>: {self.e_str}") + try: + local_vars = {'self': self, 'cp': cp} + self.optz = self._evaluate_expression(self.code, local_vars=local_vars) + except Exception as e: + raise Exception(f"Error in evaluating Objective <{self.name}>.\n{e}") + return True + + def _evaluate_expression(self, code, local_vars=None): + """ + Helper method to evaluate the expression code. + + Parameters + ---------- + code : str + The code string representing the expression. + + Returns + ------- + cp.Expression + The evaluated cvxpy expression. + """ + return eval(code, {}, local_vars) + + def __repr__(self): + return f"{self.class_name}: {self.name} [{self.sense.upper()}]" diff --git a/ams/opt/omodel.py b/ams/opt/omodel.py index 67f098d0..8e92005c 100644 --- a/ams/opt/omodel.py +++ b/ams/opt/omodel.py @@ -1,1094 +1,19 @@ """ -Module for optimization modeling. +Module for optimization OModel. """ import logging -from typing import Any, Optional, Union +from typing import Any from collections import OrderedDict -import re -import numpy as np - -from andes.core.common import Config from andes.utils.misc import elapsed import cvxpy as cp -from ams.utils import pretty_long_message -from ams.shared import sps, _prefix, _max_length - -logger = logging.getLogger(__name__) - - -def ensure_symbols(func): - """ - Decorator to ensure that symbols are generated before parsing. - If not, it runs self.rtn.syms.generate_symbols(). - - Designed to be used on the `parse` method of the optimization elements (`OptzBase`) - and optimization model (`OModel`), i.e., `Var`, `Param`, `Constraint`, `Objective`, - and `ExpressionCalc`. - - Parsing before symbol generation can give wrong results. Ensure that symbols - are generated before calling the `parse` method. - """ - - def wrapper(self, *args, **kwargs): - if not self.rtn._syms: - logger.debug(f"<{self.rtn.class_name}> symbols are not generated yet. Generating now...") - self.rtn.syms.generate_symbols() - return func(self, *args, **kwargs) - return wrapper - - -def ensure_mats_and_parsed(func): - """ - Decorator to ensure that system matrices are built and the OModel is parsed - before evaluation. If not, it runs the necessary methods to initialize them. - - Designed to be used on the `evaluate` method of optimization elements (`OptzBase`) - and the optimization model (`OModel`), i.e., `Var`, `Param`, `Constraint`, `Objective`, - and `ExpressionCalc`. - - Evaluation before building matrices and parsing the OModel can lead to errors. Ensure that - system matrices are built and the OModel is parsed before calling the `evaluate` method. - """ - - def wrapper(self, *args, **kwargs): - try: - if not self.rtn.system.mats.initialized: - logger.debug("System matrices are not built yet. Building now...") - self.rtn.system.mats.build() - if isinstance(self, (OptzBase, Var, Param, Constraint, Objective)): - if not self.om.parsed: - logger.debug("OModel is not parsed yet. Parsing now...") - self.om.parse() - elif isinstance(self, OModel): - if not self.parsed: - logger.debug("OModel is not parsed yet. Parsing now...") - self.parse() - except Exception as e: - logger.error(f"Error during initialization or parsing: {e}") - raise - return func(self, *args, **kwargs) - return wrapper - - -class OptzBase: - """ - Base class for optimization elements. - Ensure that symbols are generated before calling the `parse` method. Parsing - before symbol generation can lead to incorrect results. - - Parameters - ---------- - name : str, optional - Name of the optimization element. - info : str, optional - Descriptive information about the optimization element. - unit : str, optional - Unit of measurement for the optimization element. - - Attributes - ---------- - rtn : ams.routines.Routine - The owner routine instance. - """ - - def __init__(self, - name: Optional[str] = None, - info: Optional[str] = None, - unit: Optional[str] = None, - ): - self.om = None - self.name = name - self.info = info - self.unit = unit - self.is_disabled = False - self.rtn = None - self.optz = None # corresponding optimization element - self.code = None - - @ensure_symbols - def parse(self): - """ - Parse the object. - """ - raise NotImplementedError - - @ensure_mats_and_parsed - def evaluate(self): - """ - Evaluate the object. - """ - raise NotImplementedError - - @property - def class_name(self): - """ - Return the class name - """ - return self.__class__.__name__ - - @property - def n(self): - """ - Return the number of elements. - """ - if self.owner is None: - return len(self.v) - else: - return self.owner.n - - @property - def shape(self): - """ - Return the shape. - """ - try: - return self.om.__dict__[self.name].shape - except KeyError: - logger.warning('Shape info is not ready before initialization.') - return None - - @property - def size(self): - """ - Return the size. - """ - if self.rtn.initialized: - return self.om.__dict__[self.name].size - else: - logger.warning(f'Routine <{self.rtn.class_name}> is not initialized yet.') - return None - - -class ExpressionCalc(OptzBase): - """ - Class for calculating expressions. - - Note that `ExpressionCalc` is not a CVXPY expression, and it should not be used - in the optimization model. - It is used to calculate expression values **after** successful optimization. - """ - - def __init__(self, - name: Optional[str] = None, - info: Optional[str] = None, - unit: Optional[str] = None, - e_str: Optional[str] = None, - model: Optional[Any] = None, - src: Optional[str] = None, - ): - OptzBase.__init__(self, name=name, info=info, unit=unit) - self.optz = None - self.e_str = e_str - self.code = None - self.model = model - self.owner = None - self.src = src - self.is_group = False - - def get_idx(self): - if self.is_group: - return self.owner.get_idx() - elif self.owner is None: - logger.info(f'ExpressionCalc <{self.name}> has no owner.') - return None - else: - return self.owner.idx.v - - @ensure_symbols - def parse(self): - """ - Parse the Expression. - """ - # parse the expression str - sub_map = self.om.rtn.syms.sub_map - code_expr = self.e_str - for pattern, replacement in sub_map.items(): - try: - code_expr = re.sub(pattern, replacement, code_expr) - except Exception as e: - raise Exception(f"Error in parsing expr <{self.name}>.\n{e}") - # store the parsed expression str code - self.code = code_expr - msg = f" - ExpressionCalc <{self.name}>: {self.e_str}" - logger.debug(pretty_long_message(msg, _prefix, max_length=_max_length)) - return True - - @ensure_mats_and_parsed - def evaluate(self): - """ - Evaluate the expression. - """ - msg = f" - Expression <{self.name}>: {self.code}" - logger.debug(pretty_long_message(msg, _prefix, max_length=_max_length)) - try: - local_vars = {'self': self, 'np': np, 'cp': cp} - self.optz = self._evaluate_expression(self.code, local_vars=local_vars) - except Exception as e: - raise Exception(f"Error in evaluating ExpressionCalc <{self.name}>.\n{e}") - return True - - def _evaluate_expression(self, code, local_vars=None): - """ - Helper method to evaluate the expression code. - - Parameters - ---------- - code : str - The code string representing the expression. - - Returns - ------- - cp.Expression - The evaluated cvxpy expression. - """ - return eval(code, {}, local_vars) - - @property - def v(self): - """ - Return the CVXPY expression value. - """ - if self.optz is None: - return None - else: - return self.optz.value - - @property - def e(self): - """ - Return the calculated expression value. - """ - if self.code is None: - logger.info(f"ExpressionCalc <{self.name}> is not parsed yet.") - return None - - val_map = self.om.rtn.syms.val_map - code = self.code - for pattern, replacement in val_map.items(): - try: - code = re.sub(pattern, replacement, code) - except TypeError as e: - raise TypeError(e) - - try: - logger.debug(pretty_long_message(f"Value code: {code}", - _prefix, max_length=_max_length)) - local_vars = {'self': self, 'np': np, 'cp': cp, 'val_map': val_map} - return self._evaluate_expression(code, local_vars) - except Exception as e: - logger.error(f"Error in calculating expr <{self.name}>.\n{e}") - return None - - def __repr__(self): - return f'{self.__class__.__name__}: {self.name}' - - -class Expression(OptzBase): - """ - Base class for expressions used in a routine. - """ - - def __init__(self, - name: Optional[str] = None, - info: Optional[str] = None, - e_str: Optional[str] = None, - ): - OptzBase.__init__(self, name=name, info=info) - self.e_str = e_str - self.optz = None - self.code = None - - @ensure_symbols - def parse(self): - """ - Parse the expression. - - Returns - ------- - bool - Returns True if the parsing is successful, False otherwise. - """ - sub_map = self.om.rtn.syms.sub_map - code_expr = self.e_str - for pattern, replacement in sub_map.items(): - try: - code_expr = re.sub(pattern, replacement, code_expr) - except Exception as e: - raise Exception(f"Error in parsing expr <{self.name}>.\n{e}") - self.code = code_expr - msg = f" - Expression <{self.name}>: {self.e_str}" - logger.debug(pretty_long_message(msg, _prefix, max_length=_max_length)) - return True - - @ensure_mats_and_parsed - def evaluate(self): - """ - Evaluate the expression. - - Returns - ------- - bool - Returns True if the evaluation is successful, False otherwise. - """ - msg = f" - Expression <{self.name}>: {self.code}" - logger.debug(pretty_long_message(msg, _prefix, max_length=_max_length)) - try: - local_vars = {'self': self, 'np': np, 'cp': cp} - self.optz = self._evaluate_expression(self.code, local_vars=local_vars) - except Exception as e: - raise Exception(f"Error in evaluating Expression <{self.name}>.\n{e}") - return True - - def _evaluate_expression(self, code, local_vars=None): - """ - Helper method to evaluate the expression code. - - Parameters - ---------- - code : str - The code string representing the expression. - - Returns - ------- - cp.Expression - The evaluated cvxpy expression. - """ - return eval(code, {}, local_vars) - - @property - def v(self): - """ - Return the CVXPY expression value. - """ - if self.optz is None: - return None - else: - return self.optz.value - - @property - def shape(self): - """ - Return the shape. - """ - try: - return self.om.__dict__[self.name].shape - except KeyError: - logger.warning('Shape info is not ready before initialization.') - return None - - @property - def size(self): - """ - Return the size. - """ - if self.rtn.initialized: - return self.om.__dict__[self.name].size - else: - logger.warning(f'Routine <{self.rtn.class_name}> is not initialized yet.') - return None - - @property - def e(self): - """ - Return the calculated expression value. - """ - if self.code is None: - logger.info(f"Expression <{self.name}> is not parsed yet.") - return None - - val_map = self.om.rtn.syms.val_map - code = self.code - for pattern, replacement in val_map.items(): - try: - code = re.sub(pattern, replacement, code) - except TypeError as e: - raise TypeError(e) - - try: - logger.debug(pretty_long_message(f"Value code: {code}", - _prefix, max_length=_max_length)) - local_vars = {'self': self, 'np': np, 'cp': cp, 'val_map': val_map} - return self._evaluate_expression(code, local_vars) - except Exception as e: - logger.error(f"Error in calculating expr <{self.name}>.\n{e}") - return None - - def __repr__(self): - return f'{self.__class__.__name__}: {self.name}' - - -class Param(OptzBase): - """ - Base class for parameters used in a routine. - - Parameters - ---------- - no_parse: bool, optional - True to skip parsing the parameter. - nonneg: bool, optional - True to set the parameter as non-negative. - nonpos: bool, optional - True to set the parameter as non-positive. - cplx: bool, optional - True to set the parameter as complex, avoiding the use of `complex`. - imag: bool, optional - True to set the parameter as imaginary. - symmetric: bool, optional - True to set the parameter as symmetric. - diag: bool, optional - True to set the parameter as diagonal. - hermitian: bool, optional - True to set the parameter as hermitian. - boolean: bool, optional - True to set the parameter as boolean. - integer: bool, optional - True to set the parameter as integer. - pos: bool, optional - True to set the parameter as positive. - neg: bool, optional - True to set the parameter as negative. - sparse: bool, optional - True to set the parameter as sparse. - """ - - def __init__(self, - name: Optional[str] = None, - info: Optional[str] = None, - unit: Optional[str] = None, - no_parse: Optional[bool] = False, - nonneg: Optional[bool] = False, - nonpos: Optional[bool] = False, - cplx: Optional[bool] = False, - imag: Optional[bool] = False, - symmetric: Optional[bool] = False, - diag: Optional[bool] = False, - hermitian: Optional[bool] = False, - boolean: Optional[bool] = False, - integer: Optional[bool] = False, - pos: Optional[bool] = False, - neg: Optional[bool] = False, - sparse: Optional[bool] = False, - ): - OptzBase.__init__(self, name=name, info=info, unit=unit) - self.no_parse = no_parse # True to skip parsing the parameter - self.sparse = sparse - - self.config = Config(name=self.class_name) # `config` that can be exported - - self.config.add(OrderedDict((('nonneg', nonneg), - ('nonpos', nonpos), - ('complex', cplx), - ('imag', imag), - ('symmetric', symmetric), - ('diag', diag), - ('hermitian', hermitian), - ('boolean', boolean), - ('integer', integer), - ('pos', pos), - ('neg', neg), - ))) - - @ensure_symbols - def parse(self): - """ - Parse the parameter. - - Returns - ------- - bool - Returns True if the parsing is successful, False otherwise. - """ - sub_map = self.om.rtn.syms.sub_map - code_param = "param(**config)" - for pattern, replacement, in sub_map.items(): - try: - code_param = re.sub(pattern, replacement, code_param) - except Exception as e: - raise Exception(f"Error in parsing param <{self.name}>.\n{e}") - self.code = code_param - return True - - @ensure_mats_and_parsed - def evaluate(self): - """ - Evaluate the parameter. - """ - if self.no_parse: - return True - - config = self.config.as_dict() - try: - msg = f"Parameter <{self.name}> is set as sparse, " - msg += "but the value is not sparse." - if self.sparse: - self.v = sps.csr_matrix(self.v) - - # Create the cvxpy.Parameter object - if isinstance(self.v, np.ndarray): - self.optz = cp.Parameter(shape=self.v.shape, **config) - else: - self.optz = cp.Parameter(**config) - self.optz.value = self.v - except ValueError: - msg = f"Parameter <{self.name}> has non-numeric value, " - msg += "set `no_parse=True`." - logger.warning(msg) - self.no_parse = True - return True - except Exception as e: - raise Exception(f"Error in evaluating Param <{self.name}>.\n{e}") - return True - - def update(self): - """ - Update the Parameter value. - """ - # NOTE: skip no_parse parameters - if self.optz is None: - return None - self.optz.value = self.v - return True - - def __repr__(self): - return f'{self.__class__.__name__}: {self.name}' - - -class Var(OptzBase): - """ - Base class for variables used in a routine. - - When `horizon` is provided, the variable will be expanded to a matrix, - where rows are indexed by the source variable index and columns are - indexed by the horizon index. - - Parameters - ---------- - info : str, optional - Descriptive information - unit : str, optional - Unit - tex_name : str - LaTeX-formatted variable symbol. Defaults to the value of ``name``. - name : str, optional - Variable name. One should typically assigning the name directly because - it will be automatically assigned by the model. The value of ``name`` - will be the symbol name to be used in expressions. - src : str, optional - Source variable name. Defaults to the value of ``name``. - model : str, optional - Name of the owner model or group. - horizon : ams.routines.RParam, optional - Horizon idx. - nonneg : bool, optional - Non-negative variable - nonpos : bool, optional - Non-positive variable - cplx : bool, optional - Complex variable - imag : bool, optional - Imaginary variable - symmetric : bool, optional - Symmetric variable - diag : bool, optional - Diagonal variable - psd : bool, optional - Positive semi-definite variable - nsd : bool, optional - Negative semi-definite variable - hermitian : bool, optional - Hermitian variable - boolean : bool, optional - Boolean variable - integer : bool, optional - Integer variable - pos : bool, optional - Positive variable - neg : bool, optional - Negative variable - - Attributes - ---------- - a : np.ndarray - Variable address. - _v : np.ndarray - Local-storage of the variable value. - rtn : ams.routines.Routine - The owner routine instance. - """ - - def __init__(self, - name: Optional[str] = None, - tex_name: Optional[str] = None, - info: Optional[str] = None, - src: Optional[str] = None, - unit: Optional[str] = None, - model: Optional[str] = None, - shape: Optional[Union[tuple, int]] = None, - v0: Optional[str] = None, - horizon=None, - nonneg: Optional[bool] = False, - nonpos: Optional[bool] = False, - cplx: Optional[bool] = False, - imag: Optional[bool] = False, - symmetric: Optional[bool] = False, - diag: Optional[bool] = False, - psd: Optional[bool] = False, - nsd: Optional[bool] = False, - hermitian: Optional[bool] = False, - boolean: Optional[bool] = False, - integer: Optional[bool] = False, - pos: Optional[bool] = False, - neg: Optional[bool] = False, - ): - self.name = name - self.info = info - self.unit = unit - - self.tex_name = tex_name if tex_name else name - # variable internal index inside a model (assigned in run time) - self.id = None - OptzBase.__init__(self, name=name, info=info, unit=unit) - self.src = src - self.is_group = False - self.model = model # indicate if this variable is a group variable - self.owner = None # instance of the owner model or group - self.v0 = v0 - self.horizon = horizon - self._shape = shape - self._v = None - self.a: np.ndarray = np.array([], dtype=int) - - self.config = Config(name=self.class_name) # `config` that can be exported - - self.config.add(OrderedDict((('nonneg', nonneg), - ('nonpos', nonpos), - ('complex', cplx), - ('imag', imag), - ('symmetric', symmetric), - ('diag', diag), - ('psd', psd), - ('nsd', nsd), - ('hermitian', hermitian), - ('boolean', boolean), - ('integer', integer), - ('pos', pos), - ('neg', neg), - ))) - - @property - def v(self): - """ - Return the CVXPY variable value. - """ - if self.optz is None: - return None - if self.optz.value is None: - try: - shape = self.optz.shape - return np.zeros(shape) - except AttributeError: - return None - else: - return self.optz.value - - @v.setter - def v(self, value): - if self.optz is None: - logger.info(f"Variable <{self.name}> is not initialized yet.") - else: - self.optz.value = value - - def get_idx(self): - if self.is_group: - return self.owner.get_idx() - elif self.owner is None: - logger.info(f'Variable <{self.name}> has no owner.') - return None - else: - return self.owner.idx.v - - @ensure_symbols - def parse(self): - """ - Parse the variable. - """ - sub_map = self.om.rtn.syms.sub_map - # NOTE: number of rows is the size of the source variable - if self.owner is not None: - nr = self.owner.n - nc = 0 - if self.horizon: - # NOTE: numer of columns is the horizon if exists - nc = int(self.horizon.n) - shape = (nr, nc) - else: - shape = (nr,) - elif isinstance(self._shape, int): - shape = (self._shape,) - nr = shape - nc = 0 - elif isinstance(self._shape, tuple): - shape = self._shape - nr = shape[0] - nc = shape[1] if len(shape) > 1 else 0 - else: - raise ValueError(f"Invalid shape {self._shape}.") - code_var = f"var({shape}, **config)" - logger.debug(f" - Var <{self.name}>: {self.code}") - for pattern, replacement, in sub_map.items(): - try: - code_var = re.sub(pattern, replacement, code_var) - except Exception as e: - raise Exception(f"Error in parsing var <{self.name}>.\n{e}") - self.code = code_var - return True - - @ensure_mats_and_parsed - def evaluate(self): - """ - Evaluate the variable. - - Returns - ------- - bool - Returns True if the evaluation is successful, False otherwise. - """ - # NOTE: in CVXPY, Config only allow lower case letters - config = {} # used in `self.code` - for k, v in self.config.as_dict().items(): - if k == 'psd': - config['PSD'] = v - elif k == 'nsd': - config['NSD'] = v - elif k == 'bool': - config['boolean'] = v - else: - config[k] = v - msg = f" - Var <{self.name}>: {self.code}" - logger.debug(pretty_long_message(msg, _prefix, max_length=_max_length)) - try: - local_vars = {'self': self, 'config': config, 'cp': cp} - self.optz = eval(self.code, {}, local_vars) - except Exception as e: - raise Exception(f"Error in evaluating Var <{self.name}>.\n{e}") - return True - - def __repr__(self): - return f'{self.__class__.__name__}: {self.owner.__class__.__name__}.{self.name}' - - -class Constraint(OptzBase): - """ - Base class for constraints. - - This class is used as a template for defining constraints. Each - instance of this class represents a single constraint. - - Parameters - ---------- - name : str, optional - A user-defined name for the constraint. - e_str : str, optional - A mathematical expression representing the constraint. - info : str, optional - Additional informational text about the constraint. - is_eq : str, optional - Flag indicating if the constraint is an equality constraint. False indicates - an inequality constraint in the form of `<= 0`. - - Attributes - ---------- - is_disabled : bool - Flag indicating if the constraint is disabled, False by default. - rtn : ams.routines.Routine - The owner routine instance. - is_disabled : bool, optional - Flag indicating if the constraint is disabled, False by default. - dual : float, optional - The dual value of the constraint. - code : str, optional - The code string for the constraint - """ - - def __init__(self, - name: Optional[str] = None, - e_str: Optional[str] = None, - info: Optional[str] = None, - is_eq: Optional[bool] = False, - ): - OptzBase.__init__(self, name=name, info=info) - self.e_str = e_str - self.is_eq = is_eq - self.is_disabled = False - self.dual = None - self.code = None - - @ensure_symbols - def parse(self): - """ - Parse the constraint. - """ - # parse the expression str - sub_map = self.om.rtn.syms.sub_map - code_constr = self.e_str - for pattern, replacement in sub_map.items(): - try: - code_constr = re.sub(pattern, replacement, code_constr) - except TypeError as e: - raise TypeError(f"Error in parsing constr <{self.name}>.\n{e}") - # parse the constraint type - code_constr += " == 0" if self.is_eq else " <= 0" - # store the parsed expression str code - self.code = code_constr - msg = f" - Constr <{self.name}>: {self.e_str}" - logger.debug(pretty_long_message(msg, _prefix, max_length=_max_length)) - return True - - def _evaluate_expression(self, code, local_vars=None): - """ - Helper method to evaluate the expression code. - - Parameters - ---------- - code : str - The code string representing the expression. - - Returns - ------- - cp.Expression - The evaluated cvxpy expression. - """ - return eval(code, {}, local_vars) - - @ensure_mats_and_parsed - def evaluate(self): - """ - Evaluate the constraint. - """ - msg = f" - Constr <{self.name}>: {self.code}" - logger.debug(pretty_long_message(msg, _prefix, max_length=_max_length)) - try: - local_vars = {'self': self, 'cp': cp, 'sub_map': self.om.rtn.syms.val_map} - self.optz = self._evaluate_expression(self.code, local_vars=local_vars) - except Exception as e: - raise Exception(f"Error in evaluating Constraint <{self.name}>.\n{e}") - - def __repr__(self): - enabled = 'OFF' if self.is_disabled else 'ON' - out = f"{self.class_name}: {self.name} [{enabled}]" - return out - - @property - def e(self): - """ - Return the calculated constraint LHS value. - Note that `v` should be used primarily as it is obtained - from the solver directly. - - `e` is for debugging purpose. For a successfully solved problem, - `e` should equal to `v`. However, when a problem is infeasible - or unbounded, `e` can be used to check the constraint LHS value. - """ - if self.code is None: - logger.info(f"Constraint <{self.name}> is not parsed yet.") - return None - - val_map = self.om.rtn.syms.val_map - code = self.code - for pattern, replacement in val_map.items(): - try: - code = re.sub(pattern, replacement, code) - except TypeError as e: - raise TypeError(e) - - try: - logger.debug(pretty_long_message(f"Value code: {code}", - _prefix, max_length=_max_length)) - local_vars = {'self': self, 'np': np, 'cp': cp, 'val_map': val_map} - return self._evaluate_expression(code, local_vars) - except Exception as e: - logger.error(f"Error in calculating constr <{self.name}>.\n{e}") - return None - - @property - def v(self): - """ - Return the CVXPY constraint LHS value. - """ - if self.optz is None: - return None - if self.optz._expr.value is None: - try: - shape = self._expr.shape - return np.zeros(shape) - except AttributeError: - return None - else: - return self.optz._expr.value - - @v.setter - def v(self, value): - raise AttributeError("Cannot set the value of the constraint.") - - -class Objective(OptzBase): - """ - Base class for objective functions. - - This class serves as a template for defining objective functions. Each - instance of this class represents a single objective function that can - be minimized or maximized depending on the sense ('min' or 'max'). - - Parameters - ---------- - name : str, optional - A user-defined name for the objective function. - e_str : str, optional - A mathematical expression representing the objective function. - info : str, optional - Additional informational text about the objective function. - sense : str, optional - The sense of the objective function, default to 'min'. - `min` for minimization and `max` for maximization. +from ams.opt.optbase import ensure_symbols, ensure_mats_and_parsed - Attributes - ---------- - v : NoneType - The value of the objective function. It needs to be set through - computation. - rtn : ams.routines.Routine - The owner routine instance. - code : str - The code string for the objective function. - """ - - def __init__(self, - name: Optional[str] = None, - e_str: Optional[str] = None, - info: Optional[str] = None, - unit: Optional[str] = None, - sense: Optional[str] = 'min'): - OptzBase.__init__(self, name=name, info=info, unit=unit) - self.e_str = e_str - self.sense = sense - self.code = None - - @property - def e(self): - """ - Return the calculated objective value. - - Note that `v` should be used primarily as it is obtained - from the solver directly. - - `e` is for debugging purpose. For a successfully solved problem, - `e` should equal to `v`. However, when a problem is infeasible - or unbounded, `e` can be used to check the objective value. - """ - if self.code is None: - logger.info(f"Objective <{self.name}> is not parsed yet.") - return None - val_map = self.om.rtn.syms.val_map - code = self.code - for pattern, replacement in val_map.items(): - try: - code = re.sub(pattern, replacement, code) - except TypeError as e: - logger.error(f"Error in parsing value for obj <{self.name}>.") - raise e - - try: - logger.debug(pretty_long_message(f"Value code: {code}", - _prefix, max_length=_max_length)) - local_vars = {'self': self, 'np': np, 'cp': cp, 'val_map': val_map} - return self._evaluate_expression(code, local_vars) - except Exception as e: - logger.error(f"Error in calculating obj <{self.name}>.\n{e}") - return None - - @property - def v(self): - """ - Return the CVXPY objective value. - """ - if self.optz is None: - return None - else: - return self.optz.value - - @v.setter - def v(self, value): - raise AttributeError("Cannot set the value of the objective function.") - - @ensure_symbols - def parse(self): - """ - Parse the objective function. - - Returns - ------- - bool - Returns True if the parsing is successful, False otherwise. - """ - # parse the expression str - sub_map = self.om.rtn.syms.sub_map - code_obj = self.e_str - for pattern, replacement, in sub_map.items(): - try: - code_obj = re.sub(pattern, replacement, code_obj) - except Exception as e: - raise Exception(f"Error in parsing obj <{self.name}>.\n{e}") - # store the parsed expression str code - self.code = code_obj - if self.sense not in ['min', 'max']: - raise ValueError(f'Objective sense {self.sense} is not supported.') - sense = 'cp.Minimize' if self.sense == 'min' else 'cp.Maximize' - self.code = f"{sense}({code_obj})" - msg = f" - Objective <{self.name}>: {self.code}" - logger.debug(pretty_long_message(msg, _prefix, max_length=_max_length)) - return True - - @ensure_mats_and_parsed - def evaluate(self): - """ - Evaluate the objective function. - - Returns - ------- - bool - Returns True if the evaluation is successful, False otherwise. - """ - logger.debug(f" - Objective <{self.name}>: {self.e_str}") - try: - local_vars = {'self': self, 'cp': cp} - self.optz = self._evaluate_expression(self.code, local_vars=local_vars) - except Exception as e: - raise Exception(f"Error in evaluating Objective <{self.name}>.\n{e}") - return True - - def _evaluate_expression(self, code, local_vars=None): - """ - Helper method to evaluate the expression code. - - Parameters - ---------- - code : str - The code string representing the expression. - - Returns - ------- - cp.Expression - The evaluated cvxpy expression. - """ - return eval(code, {}, local_vars) - - def __repr__(self): - return f"{self.class_name}: {self.name} [{self.sense.upper()}]" +logger = logging.getLogger(__name__) class OModel: diff --git a/ams/opt/optbase.py b/ams/opt/optbase.py new file mode 100644 index 00000000..c9773826 --- /dev/null +++ b/ams/opt/optbase.py @@ -0,0 +1,152 @@ +""" +Module for optimization base classes. +""" +import logging + +from typing import Optional + + +logger = logging.getLogger(__name__) + + +def ensure_symbols(func): + """ + Decorator to ensure that symbols are generated before parsing. + If not, it runs self.rtn.syms.generate_symbols(). + + Designed to be used on the `parse` method of the optimization elements (`OptzBase`) + and optimization model (`OModel`), i.e., `Var`, `Param`, `Constraint`, `Objective`, + and `ExpressionCalc`. + + Parsing before symbol generation can give wrong results. Ensure that symbols + are generated before calling the `parse` method. + """ + + def wrapper(self, *args, **kwargs): + if not self.rtn._syms: + logger.debug(f"<{self.rtn.class_name}> symbols are not generated yet. Generating now...") + self.rtn.syms.generate_symbols() + return func(self, *args, **kwargs) + return wrapper + + +def ensure_mats_and_parsed(func): + """ + Decorator to ensure that system matrices are built and the OModel is parsed + before evaluation. If not, it runs the necessary methods to initialize them. + + Designed to be used on the `evaluate` method of optimization elements (`OptzBase`) + and the optimization model (`OModel`), i.e., `Var`, `Param`, `Constraint`, `Objective`, + and `ExpressionCalc`. + + Evaluation before building matrices and parsing the OModel can lead to errors. Ensure that + system matrices are built and the OModel is parsed before calling the `evaluate` method. + """ + + def wrapper(self, *args, **kwargs): + try: + if not self.rtn.system.mats.initialized: + logger.debug("System matrices are not built yet. Building now...") + self.rtn.system.mats.build() + if isinstance(self, (OptzBase)): + if not self.om.parsed: + logger.debug("OModel is not parsed yet. Parsing now...") + self.om.parse() + else: + if not self.parsed: + logger.debug("OModel is not parsed yet. Parsing now...") + self.parse() + except Exception as e: + logger.error(f"Error during initialization or parsing: {e}") + raise + return func(self, *args, **kwargs) + return wrapper + + +class OptzBase: + """ + Base class for optimization elements. + Ensure that symbols are generated before calling the `parse` method. Parsing + before symbol generation can lead to incorrect results. + + Parameters + ---------- + name : str, optional + Name of the optimization element. + info : str, optional + Descriptive information about the optimization element. + unit : str, optional + Unit of measurement for the optimization element. + + Attributes + ---------- + rtn : ams.routines.Routine + The owner routine instance. + """ + + def __init__(self, + name: Optional[str] = None, + info: Optional[str] = None, + unit: Optional[str] = None, + ): + self.om = None + self.name = name + self.info = info + self.unit = unit + self.is_disabled = False + self.rtn = None + self.optz = None # corresponding optimization element + self.code = None + + @ensure_symbols + def parse(self): + """ + Parse the object. + """ + raise NotImplementedError + + @ensure_mats_and_parsed + def evaluate(self): + """ + Evaluate the object. + """ + raise NotImplementedError + + @property + def class_name(self): + """ + Return the class name + """ + return self.__class__.__name__ + + @property + def n(self): + """ + Return the number of elements. + """ + if self.owner is None: + return len(self.v) + else: + return self.owner.n + + @property + def shape(self): + """ + Return the shape. + """ + try: + return self.om.__dict__[self.name].shape + except KeyError: + logger.warning('Shape info is not ready before initialization.') + return None + + @property + def size(self): + """ + Return the size. + """ + if self.rtn.initialized: + return self.om.__dict__[self.name].size + else: + logger.warning(f'Routine <{self.rtn.class_name}> is not initialized yet.') + return None diff --git a/ams/opt/param.py b/ams/opt/param.py new file mode 100644 index 00000000..f8860fe0 --- /dev/null +++ b/ams/opt/param.py @@ -0,0 +1,156 @@ +""" +Module for optimization Param. +""" +import logging + +from typing import Optional +from collections import OrderedDict +import re + +import numpy as np + +from andes.core.common import Config + +import cvxpy as cp + +from ams.shared import sps + +from ams.opt import OptzBase, ensure_symbols, ensure_mats_and_parsed + +logger = logging.getLogger(__name__) + + +class Param(OptzBase): + """ + Base class for parameters used in a routine. + + Parameters + ---------- + no_parse: bool, optional + True to skip parsing the parameter. + nonneg: bool, optional + True to set the parameter as non-negative. + nonpos: bool, optional + True to set the parameter as non-positive. + cplx: bool, optional + True to set the parameter as complex, avoiding the use of `complex`. + imag: bool, optional + True to set the parameter as imaginary. + symmetric: bool, optional + True to set the parameter as symmetric. + diag: bool, optional + True to set the parameter as diagonal. + hermitian: bool, optional + True to set the parameter as hermitian. + boolean: bool, optional + True to set the parameter as boolean. + integer: bool, optional + True to set the parameter as integer. + pos: bool, optional + True to set the parameter as positive. + neg: bool, optional + True to set the parameter as negative. + sparse: bool, optional + True to set the parameter as sparse. + """ + + def __init__(self, + name: Optional[str] = None, + info: Optional[str] = None, + unit: Optional[str] = None, + no_parse: Optional[bool] = False, + nonneg: Optional[bool] = False, + nonpos: Optional[bool] = False, + cplx: Optional[bool] = False, + imag: Optional[bool] = False, + symmetric: Optional[bool] = False, + diag: Optional[bool] = False, + hermitian: Optional[bool] = False, + boolean: Optional[bool] = False, + integer: Optional[bool] = False, + pos: Optional[bool] = False, + neg: Optional[bool] = False, + sparse: Optional[bool] = False, + ): + OptzBase.__init__(self, name=name, info=info, unit=unit) + self.no_parse = no_parse # True to skip parsing the parameter + self.sparse = sparse + + self.config = Config(name=self.class_name) # `config` that can be exported + + self.config.add(OrderedDict((('nonneg', nonneg), + ('nonpos', nonpos), + ('complex', cplx), + ('imag', imag), + ('symmetric', symmetric), + ('diag', diag), + ('hermitian', hermitian), + ('boolean', boolean), + ('integer', integer), + ('pos', pos), + ('neg', neg), + ))) + + @ensure_symbols + def parse(self): + """ + Parse the parameter. + + Returns + ------- + bool + Returns True if the parsing is successful, False otherwise. + """ + sub_map = self.om.rtn.syms.sub_map + code_param = "param(**config)" + for pattern, replacement, in sub_map.items(): + try: + code_param = re.sub(pattern, replacement, code_param) + except Exception as e: + raise Exception(f"Error in parsing param <{self.name}>.\n{e}") + self.code = code_param + return True + + @ensure_mats_and_parsed + def evaluate(self): + """ + Evaluate the parameter. + """ + if self.no_parse: + return True + + config = self.config.as_dict() + try: + msg = f"Parameter <{self.name}> is set as sparse, " + msg += "but the value is not sparse." + if self.sparse: + self.v = sps.csr_matrix(self.v) + + # Create the cvxpy.Parameter object + if isinstance(self.v, np.ndarray): + self.optz = cp.Parameter(shape=self.v.shape, **config) + else: + self.optz = cp.Parameter(**config) + self.optz.value = self.v + except ValueError: + msg = f"Parameter <{self.name}> has non-numeric value, " + msg += "set `no_parse=True`." + logger.warning(msg) + self.no_parse = True + return True + except Exception as e: + raise Exception(f"Error in evaluating Param <{self.name}>.\n{e}") + return True + + def update(self): + """ + Update the Parameter value. + """ + # NOTE: skip no_parse parameters + if self.optz is None: + return None + self.optz.value = self.v + return True + + def __repr__(self): + return f'{self.__class__.__name__}: {self.name}' diff --git a/ams/opt/var.py b/ams/opt/var.py new file mode 100644 index 00000000..39d8d512 --- /dev/null +++ b/ams/opt/var.py @@ -0,0 +1,245 @@ +""" +Module for optimization Var. +""" +import logging + +from typing import Optional, Union +from collections import OrderedDict +import re + +import numpy as np + +from andes.core.common import Config + +import cvxpy as cp + +from ams.utils import pretty_long_message +from ams.shared import _prefix, _max_length + +from ams.opt import OptzBase, ensure_symbols, ensure_mats_and_parsed + +logger = logging.getLogger(__name__) + + +class Var(OptzBase): + """ + Base class for variables used in a routine. + + When `horizon` is provided, the variable will be expanded to a matrix, + where rows are indexed by the source variable index and columns are + indexed by the horizon index. + + Parameters + ---------- + info : str, optional + Descriptive information + unit : str, optional + Unit + tex_name : str + LaTeX-formatted variable symbol. Defaults to the value of ``name``. + name : str, optional + Variable name. One should typically assigning the name directly because + it will be automatically assigned by the model. The value of ``name`` + will be the symbol name to be used in expressions. + src : str, optional + Source variable name. Defaults to the value of ``name``. + model : str, optional + Name of the owner model or group. + horizon : ams.routines.RParam, optional + Horizon idx. + nonneg : bool, optional + Non-negative variable + nonpos : bool, optional + Non-positive variable + cplx : bool, optional + Complex variable + imag : bool, optional + Imaginary variable + symmetric : bool, optional + Symmetric variable + diag : bool, optional + Diagonal variable + psd : bool, optional + Positive semi-definite variable + nsd : bool, optional + Negative semi-definite variable + hermitian : bool, optional + Hermitian variable + boolean : bool, optional + Boolean variable + integer : bool, optional + Integer variable + pos : bool, optional + Positive variable + neg : bool, optional + Negative variable + + Attributes + ---------- + a : np.ndarray + Variable address. + _v : np.ndarray + Local-storage of the variable value. + rtn : ams.routines.Routine + The owner routine instance. + """ + + def __init__(self, + name: Optional[str] = None, + tex_name: Optional[str] = None, + info: Optional[str] = None, + src: Optional[str] = None, + unit: Optional[str] = None, + model: Optional[str] = None, + shape: Optional[Union[tuple, int]] = None, + v0: Optional[str] = None, + horizon=None, + nonneg: Optional[bool] = False, + nonpos: Optional[bool] = False, + cplx: Optional[bool] = False, + imag: Optional[bool] = False, + symmetric: Optional[bool] = False, + diag: Optional[bool] = False, + psd: Optional[bool] = False, + nsd: Optional[bool] = False, + hermitian: Optional[bool] = False, + boolean: Optional[bool] = False, + integer: Optional[bool] = False, + pos: Optional[bool] = False, + neg: Optional[bool] = False, + ): + self.name = name + self.info = info + self.unit = unit + + self.tex_name = tex_name if tex_name else name + # variable internal index inside a model (assigned in run time) + self.id = None + OptzBase.__init__(self, name=name, info=info, unit=unit) + self.src = src + self.is_group = False + self.model = model # indicate if this variable is a group variable + self.owner = None # instance of the owner model or group + self.v0 = v0 + self.horizon = horizon + self._shape = shape + self._v = None + self.a: np.ndarray = np.array([], dtype=int) + + self.config = Config(name=self.class_name) # `config` that can be exported + + self.config.add(OrderedDict((('nonneg', nonneg), + ('nonpos', nonpos), + ('complex', cplx), + ('imag', imag), + ('symmetric', symmetric), + ('diag', diag), + ('psd', psd), + ('nsd', nsd), + ('hermitian', hermitian), + ('boolean', boolean), + ('integer', integer), + ('pos', pos), + ('neg', neg), + ))) + + @property + def v(self): + """ + Return the CVXPY variable value. + """ + if self.optz is None: + return None + if self.optz.value is None: + try: + shape = self.optz.shape + return np.zeros(shape) + except AttributeError: + return None + else: + return self.optz.value + + @v.setter + def v(self, value): + if self.optz is None: + logger.info(f"Variable <{self.name}> is not initialized yet.") + else: + self.optz.value = value + + def get_idx(self): + if self.is_group: + return self.owner.get_idx() + elif self.owner is None: + logger.info(f'Variable <{self.name}> has no owner.') + return None + else: + return self.owner.idx.v + + @ensure_symbols + def parse(self): + """ + Parse the variable. + """ + sub_map = self.om.rtn.syms.sub_map + # NOTE: number of rows is the size of the source variable + if self.owner is not None: + nr = self.owner.n + nc = 0 + if self.horizon: + # NOTE: numer of columns is the horizon if exists + nc = int(self.horizon.n) + shape = (nr, nc) + else: + shape = (nr,) + elif isinstance(self._shape, int): + shape = (self._shape,) + nr = shape + nc = 0 + elif isinstance(self._shape, tuple): + shape = self._shape + nr = shape[0] + nc = shape[1] if len(shape) > 1 else 0 + else: + raise ValueError(f"Invalid shape {self._shape}.") + code_var = f"var({shape}, **config)" + logger.debug(f" - Var <{self.name}>: {self.code}") + for pattern, replacement, in sub_map.items(): + try: + code_var = re.sub(pattern, replacement, code_var) + except Exception as e: + raise Exception(f"Error in parsing var <{self.name}>.\n{e}") + self.code = code_var + return True + + @ensure_mats_and_parsed + def evaluate(self): + """ + Evaluate the variable. + + Returns + ------- + bool + Returns True if the evaluation is successful, False otherwise. + """ + # NOTE: in CVXPY, Config only allow lower case letters + config = {} # used in `self.code` + for k, v in self.config.as_dict().items(): + if k == 'psd': + config['PSD'] = v + elif k == 'nsd': + config['NSD'] = v + elif k == 'bool': + config['boolean'] = v + else: + config[k] = v + msg = f" - Var <{self.name}>: {self.code}" + logger.debug(pretty_long_message(msg, _prefix, max_length=_max_length)) + try: + local_vars = {'self': self, 'config': config, 'cp': cp} + self.optz = eval(self.code, {}, local_vars) + except Exception as e: + raise Exception(f"Error in evaluating Var <{self.name}>.\n{e}") + return True + + def __repr__(self): + return f'{self.__class__.__name__}: {self.owner.__class__.__name__}.{self.name}' diff --git a/ams/routines/acopf.py b/ams/routines/acopf.py index 3a746ffb..b9e77528 100644 --- a/ams/routines/acopf.py +++ b/ams/routines/acopf.py @@ -11,7 +11,7 @@ from ams.core.param import RParam from ams.routines.dcpf import DCPF -from ams.opt.omodel import Var, Constraint, Objective +from ams.opt import Var, Constraint, Objective logger = logging.getLogger(__name__) diff --git a/ams/routines/dcopf.py b/ams/routines/dcopf.py index 0f9c5595..66d5eb00 100644 --- a/ams/routines/dcopf.py +++ b/ams/routines/dcopf.py @@ -9,7 +9,7 @@ from ams.routines.routine import RoutineBase -from ams.opt.omodel import Var, Constraint, Objective, ExpressionCalc +from ams.opt import Var, Constraint, Objective, ExpressionCalc logger = logging.getLogger(__name__) diff --git a/ams/routines/dcpf.py b/ams/routines/dcpf.py index d0f84735..ee7f46b6 100644 --- a/ams/routines/dcpf.py +++ b/ams/routines/dcpf.py @@ -7,7 +7,7 @@ from andes.utils.misc import elapsed from ams.routines.routine import RoutineBase -from ams.opt.omodel import Var +from ams.opt import Var from ams.pypower import runpf from ams.pypower.core import ppoption diff --git a/ams/routines/dopf.py b/ams/routines/dopf.py index 2d2ef76c..1771d3cb 100644 --- a/ams/routines/dopf.py +++ b/ams/routines/dopf.py @@ -7,7 +7,7 @@ from ams.routines.dcopf import DCOPF -from ams.opt.omodel import Var, Constraint, Objective +from ams.opt import Var, Constraint, Objective class DOPF(DCOPF): diff --git a/ams/routines/ed.py b/ams/routines/ed.py index 1ea2592f..f5624d73 100644 --- a/ams/routines/ed.py +++ b/ams/routines/ed.py @@ -10,7 +10,7 @@ from ams.routines.rted import RTED, DGBase, ESD1Base -from ams.opt.omodel import Var, Constraint +from ams.opt import Var, Constraint logger = logging.getLogger(__name__) diff --git a/ams/routines/pflow.py b/ams/routines/pflow.py index 68ec9bd2..1f912765 100644 --- a/ams/routines/pflow.py +++ b/ams/routines/pflow.py @@ -11,7 +11,7 @@ from ams.core.param import RParam from ams.routines.dcpf import DCPF -from ams.opt.omodel import Var +from ams.opt import Var logger = logging.getLogger(__name__) diff --git a/ams/routines/pflow2.py b/ams/routines/pflow2.py index c7e2cd00..9025a9d1 100644 --- a/ams/routines/pflow2.py +++ b/ams/routines/pflow2.py @@ -9,7 +9,7 @@ from ams.routines.routine import RoutineBase -from ams.opt.omodel import Var, Constraint, Objective +from ams.opt import Var, Constraint, Objective logger = logging.getLogger(__name__) @@ -196,11 +196,6 @@ def __init__(self, system, config): unit='p.u.', name='vBus', tex_name=r'v_{bus}', model='Bus', src='v',) - self.Vc = Var(info='Bus voltage in complex', - unit='p.u.', - name='V', tex_name=r'V', - model='Bus', - cplx=True,) self.csb = VarSelect(info='select slack bus', name='csb', tex_name=r'c_{sb}', u=self.aBus, indexer='buss', @@ -209,6 +204,9 @@ def __init__(self, system, config): name='sbus', is_eq=True, e_str='csb@aBus',) # --- power balance --- + # self.Vc = ExpressionCalc(info='Expression of Bus voltage in complex', + # name='Vc', + # e_str='vBus dot exp(1j dot aBus)',) # Vc = vBus * exp(1j * aBus) # TODO: pout = VYV; diag(Vc) @ conj(Y) @ Vc # TODO: pin = $injected power$ diff --git a/ams/routines/routine.py b/ams/routines/routine.py index 19770008..2b19d7c5 100644 --- a/ams/routines/routine.py +++ b/ams/routines/routine.py @@ -16,7 +16,7 @@ from ams.core.symprocessor import SymProcessor from ams.core.documenter import RDocumenter from ams.core.service import RBaseService, ValueService -from ams.opt.omodel import OModel, Param, Var, Constraint, Objective, ExpressionCalc, Expression +from ams.opt import OModel, Param, Var, Constraint, Objective, ExpressionCalc, Expression from ams.shared import pd diff --git a/ams/routines/rted.py b/ams/routines/rted.py index c04ce86f..28099dc1 100644 --- a/ams/routines/rted.py +++ b/ams/routines/rted.py @@ -9,7 +9,7 @@ from ams.core.service import ZonalSum, VarSelect, NumOp, NumOpDual from ams.routines.dcopf import DCOPF -from ams.opt.omodel import Var, Constraint +from ams.opt import Var, Constraint logger = logging.getLogger(__name__) diff --git a/ams/routines/uc.py b/ams/routines/uc.py index f1b29827..92527de8 100644 --- a/ams/routines/uc.py +++ b/ams/routines/uc.py @@ -12,7 +12,7 @@ from ams.routines.rted import RTEDBase from ams.routines.ed import SRBase, MPBase, ESD1MPBase, DGBase -from ams.opt.omodel import Var, Constraint +from ams.opt import Var, Constraint logger = logging.getLogger(__name__) From beaeb4012c1e4766e3696d32a4d0c39f40c17e10 Mon Sep 17 00:00:00 2001 From: jinningwang Date: Fri, 15 Nov 2024 14:17:38 -0500 Subject: [PATCH 23/96] Update release notes --- docs/source/release-notes.rst | 1 + 1 file changed, 1 insertion(+) diff --git a/docs/source/release-notes.rst b/docs/source/release-notes.rst index 8990309a..6149889a 100644 --- a/docs/source/release-notes.rst +++ b/docs/source/release-notes.rst @@ -18,6 +18,7 @@ TODO: class `Expression`: registry in `OModel` and `Routine`, fit symbol process - Add a demo to show using `Constraint.e` for debugging - Fix `ams.opt.omodel.Param.evaluate()` when its value is a number - Improve `ams.opt.omodel.ExpressionCalc()` for efficiency +- Refactor module `ams.opt` RC1 ~~~~ From a1b1b50110a7a984120bd6e636629d15d23c3df2 Mon Sep 17 00:00:00 2001 From: jinningwang Date: Fri, 15 Nov 2024 14:22:37 -0500 Subject: [PATCH 24/96] Typo --- docs/source/release-notes.rst | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/docs/source/release-notes.rst b/docs/source/release-notes.rst index 6149889a..38eabdda 100644 --- a/docs/source/release-notes.rst +++ b/docs/source/release-notes.rst @@ -14,7 +14,7 @@ v0.9.12 (202x-xx-xx) TODO: class `Expression`: registry in `OModel` and `Routine`, fit symbol processor, fit documenter -- Refactor OModel.initialized as a property method +- Refactor `OModel.initialized` as a property method - Add a demo to show using `Constraint.e` for debugging - Fix `ams.opt.omodel.Param.evaluate()` when its value is a number - Improve `ams.opt.omodel.ExpressionCalc()` for efficiency From f8d4770be334180c1b06b1909770467f215099ec Mon Sep 17 00:00:00 2001 From: jinningwang Date: Fri, 15 Nov 2024 14:29:06 -0500 Subject: [PATCH 25/96] Typo --- examples/demonstration/demo_debug.ipynb | 6 +++--- examples/ex6.ipynb | 4 ++-- 2 files changed, 5 insertions(+), 5 deletions(-) diff --git a/examples/demonstration/demo_debug.ipynb b/examples/demonstration/demo_debug.ipynb index 0ca2948a..6f144393 100644 --- a/examples/demonstration/demo_debug.ipynb +++ b/examples/demonstration/demo_debug.ipynb @@ -4,9 +4,9 @@ "cell_type": "markdown", "metadata": {}, "source": [ - "# Using LHS Value to Debug Unsolved Problems\n", + "# Using LHS Value to Debug An Infeasible Problem\n", "\n", - "This notebook aims to show how to use the left-hand side (LHS) value (`Constraint.e`) to debug unsolved problems.\n", + "This notebook aims to show how to use the left-hand side (LHS) value (`Constraint.e`) to debug an infeasible problem.\n", "\n", "The LHS value is the value of the left-hand side of the equation. It is useful to check which constraints are violated." ] @@ -140,7 +140,7 @@ "cell_type": "markdown", "metadata": {}, "source": [ - "Then, we can lower down the generator maximum output to create an unsolvable problem." + "Then, we can lower down the generator maximum output to create an infeasible problem." ] }, { diff --git a/examples/ex6.ipynb b/examples/ex6.ipynb index 0a68d364..b150d398 100644 --- a/examples/ex6.ipynb +++ b/examples/ex6.ipynb @@ -5,14 +5,14 @@ "cell_type": "markdown", "metadata": {}, "source": [ - "# Multi-period Dispatch Simulation" + "# Multi-period Scheduling Simulation" ] }, { "cell_type": "markdown", "metadata": {}, "source": [ - "Multi-period dispatch economic dispatch (ED) and unit commitment (UC) is also available.\n", + "Multi-period economic dispatch (ED) and unit commitment (UC) are also available.\n", "\n", "In this case, we will show a 24-hour ED simulation." ] From 467644560638f7a7313f949fbd20d0c6fba92be6 Mon Sep 17 00:00:00 2001 From: jinningwang Date: Fri, 15 Nov 2024 19:58:22 -0500 Subject: [PATCH 26/96] Typo --- ams/opt/exprcalc.py | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/ams/opt/exprcalc.py b/ams/opt/exprcalc.py index 25e11230..f946f43e 100644 --- a/ams/opt/exprcalc.py +++ b/ams/opt/exprcalc.py @@ -3,7 +3,7 @@ """ import logging -from typing import Any, Optional +from typing import Optional import re import numpy as np @@ -32,7 +32,7 @@ def __init__(self, info: Optional[str] = None, unit: Optional[str] = None, e_str: Optional[str] = None, - model: Optional[Any] = None, + model: Optional[str] = None, src: Optional[str] = None, ): OptzBase.__init__(self, name=name, info=info, unit=unit) From 528c9b7e436dfaed82ce9f4f355e8cae69f28f4b Mon Sep 17 00:00:00 2001 From: jinningwang Date: Fri, 15 Nov 2024 20:40:32 -0500 Subject: [PATCH 27/96] Typo --- ams/opt/var.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/ams/opt/var.py b/ams/opt/var.py index 39d8d512..ac13a8cc 100644 --- a/ams/opt/var.py +++ b/ams/opt/var.py @@ -93,7 +93,7 @@ def __init__(self, model: Optional[str] = None, shape: Optional[Union[tuple, int]] = None, v0: Optional[str] = None, - horizon=None, + horizon: Optional[str] = None, nonneg: Optional[bool] = False, nonpos: Optional[bool] = False, cplx: Optional[bool] = False, From 233557f4b5cf82dddb2e6421854571d34253f6e7 Mon Sep 17 00:00:00 2001 From: jinningwang Date: Fri, 15 Nov 2024 20:41:03 -0500 Subject: [PATCH 28/96] Add exp to val_map replacement --- ams/core/symprocessor.py | 1 + 1 file changed, 1 insertion(+) diff --git a/ams/core/symprocessor.py b/ams/core/symprocessor.py index 2a7f9369..7643b057 100644 --- a/ams/core/symprocessor.py +++ b/ams/core/symprocessor.py @@ -89,6 +89,7 @@ def __init__(self, parent): (r'(== 0|<= 0)$', ''), # remove the comparison operator (r'cp\.(Minimize|Maximize)', r'float'), # remove cp.Minimize/Maximize (r'\bcp.\b', 'np.'), + (r'\bexp\b', 'np.exp'), ]) self.status = { From 89b2d72571b67ddae23ba9e1c85b24855fd0b7d7 Mon Sep 17 00:00:00 2001 From: jinningwang Date: Fri, 15 Nov 2024 20:41:23 -0500 Subject: [PATCH 29/96] Add more attributes to class Expression --- ams/opt/expr.py | 61 +++++++++++++++++++++++++++++++++++++++++++------ ams/system.py | 10 ++++++-- 2 files changed, 62 insertions(+), 9 deletions(-) diff --git a/ams/opt/expr.py b/ams/opt/expr.py index 72a715a1..a17b343e 100644 --- a/ams/opt/expr.py +++ b/ams/opt/expr.py @@ -26,12 +26,35 @@ class Expression(OptzBase): def __init__(self, name: Optional[str] = None, info: Optional[str] = None, + unit: Optional[str] = None, e_str: Optional[str] = None, + no_parse: Optional[bool] = False, + model: Optional[str] = None, + src: Optional[str] = None, + vtype: Optional[str] = float, + horizon: Optional[str] = None, ): - OptzBase.__init__(self, name=name, info=info) + OptzBase.__init__(self, name=name, info=info, unit=unit) self.e_str = e_str self.optz = None self.code = None + self.no_parse = no_parse + self.model = model + self.owner = None + self.src = src + self.is_group = False + self._v = None # internal value storage when no_parse is True + self.vtype = vtype # value type, used when no_parse is True + self.horizon = horizon + + def get_idx(self): + if self.is_group: + return self.owner.get_idx() + elif self.owner is None: + logger.info(f'ExpressionCalc <{self.name}> has no owner.') + return None + else: + return self.owner.idx.v @ensure_symbols def parse(self): @@ -65,6 +88,15 @@ def evaluate(self): bool Returns True if the evaluation is successful, False otherwise. """ + # TODO: when no_parse, should initialize _v + if self.no_parse: + if self.horizon is not None: + shape = (self.owner.n, self.horizon.n) + else: + shape = (self.owner.n,) + self._v = np.zeros(shape, dtype=self.vtype) + return True + msg = f" - Expression <{self.name}>: {self.code}" logger.debug(pretty_long_message(msg, _prefix, max_length=_max_length)) try: @@ -95,21 +127,36 @@ def v(self): """ Return the CVXPY expression value. """ - if self.optz is None: + if self.no_parse: + return self._v + elif self.optz is None: return None else: return self.optz.value + @v.setter + def v(self, value): + """ + Set the value. + """ + if self.no_parse: + self._v = value + else: + logger.warning('Cannot set value to an Expression that is not no_parse.') + @property def shape(self): """ Return the shape. """ - try: - return self.om.__dict__[self.name].shape - except KeyError: - logger.warning('Shape info is not ready before initialization.') - return None + if self.no_parse: + return self._v.shape + else: + try: + return self.om.__dict__[self.name].shape + except KeyError: + logger.warning('Shape info is not ready before initialization.') + return None @property def size(self): diff --git a/ams/system.py b/ams/system.py index 834827c3..789655bf 100644 --- a/ams/system.py +++ b/ams/system.py @@ -229,10 +229,13 @@ def import_types(self): def _collect_group_data(self, items): """ - Set the owner for routine attributes: ``RParam``, ``Var``, ``ExpressionCalc``, and ``RBaseService``. + Set the owner for routine attributes: `RParam`, `Var`, `ExpressionCalc`, `Expression`, + and `RBaseService`. """ for item_name, item in items.items(): - if item.model in self.groups.keys(): + if item.model is None: + continue + elif item.model in self.groups.keys(): item.is_group = True item.owner = self.groups[item.model] elif item.model in self.models.keys(): @@ -280,6 +283,9 @@ def import_routines(self): # Collect ExpressionCalcs exprc = getattr(rtn, 'exprcs') self._collect_group_data(exprc) + # Collect Expressions + expr = getattr(rtn, 'exprs') + self._collect_group_data(expr) def import_groups(self): """ From 6686232fea04bd3417a7ee8bdadfbcabb1608964 Mon Sep 17 00:00:00 2001 From: jinningwang Date: Fri, 15 Nov 2024 20:50:38 -0500 Subject: [PATCH 30/96] Add docstring to class Expression --- ams/opt/expr.py | 27 +++++++++++++++++++++++++++ 1 file changed, 27 insertions(+) diff --git a/ams/opt/expr.py b/ams/opt/expr.py index a17b343e..63a914df 100644 --- a/ams/opt/expr.py +++ b/ams/opt/expr.py @@ -21,6 +21,33 @@ class Expression(OptzBase): """ Base class for expressions used in a routine. + + When `no_parse` is True, the Expression will have its own value storage + `_v` and will not be evaluated as a CVXPY expression. + In this case, `model` should be provided to indicate the owner model or group. + + Parameters + ---------- + name : str, optional + Expression name. One should typically assigning the name directly because + it will be automatically assigned by the model. The value of ``name`` + will be the symbol name to be used in expressions. + info : str, optional + Descriptive information + unit : str, optional + Unit + e_str : str, optional + Expression string + no_parse : bool, optional + Do not parse the expression + model : str, optional + Name of the owner model or group. + src : str, optional + Source expression name + vtype : type, optional + Value type + horizon : ams.routines.RParam, optional + Horizon """ def __init__(self, From e69687c605e90e35c38867b29a89aea5798deb6d Mon Sep 17 00:00:00 2001 From: jinningwang Date: Fri, 15 Nov 2024 20:56:38 -0500 Subject: [PATCH 31/96] [WIP] PFlow2, add complex bus voltage using Expression --- ams/routines/pflow2.py | 11 +++++++---- 1 file changed, 7 insertions(+), 4 deletions(-) diff --git a/ams/routines/pflow2.py b/ams/routines/pflow2.py index 9025a9d1..19122363 100644 --- a/ams/routines/pflow2.py +++ b/ams/routines/pflow2.py @@ -9,7 +9,7 @@ from ams.routines.routine import RoutineBase -from ams.opt import Var, Constraint, Objective +from ams.opt import Var, Constraint, Objective, Expression logger = logging.getLogger(__name__) @@ -204,9 +204,12 @@ def __init__(self, system, config): name='sbus', is_eq=True, e_str='csb@aBus',) # --- power balance --- - # self.Vc = ExpressionCalc(info='Expression of Bus voltage in complex', - # name='Vc', - # e_str='vBus dot exp(1j dot aBus)',) + self.Vc = Expression(info='Bus voltage in complex', + name='Vc', unit='p.u.', + e_str='vBus dot exp(1j dot aBus)', + no_parse=True, + model='Bus', src=None, + vtype=complex,) # Vc = vBus * exp(1j * aBus) # TODO: pout = VYV; diag(Vc) @ conj(Y) @ Vc # TODO: pin = $injected power$ From d26e6778926410028ba270ff973a335082b51fda Mon Sep 17 00:00:00 2001 From: jinningwang Date: Sat, 16 Nov 2024 07:07:36 -0500 Subject: [PATCH 32/96] Fix class Expression usage --- ams/core/symprocessor.py | 18 +++++++++++------- ams/opt/expr.py | 6 +++++- ams/opt/omodel.py | 4 +++- 3 files changed, 19 insertions(+), 9 deletions(-) diff --git a/ams/core/symprocessor.py b/ams/core/symprocessor.py index 7643b057..cde68fe0 100644 --- a/ams/core/symprocessor.py +++ b/ams/core/symprocessor.py @@ -123,11 +123,19 @@ def generate_symbols(self, force_generate=False): for key in self.parent.tex_names.keys(): self.tex_names[key] = sp.symbols(self.parent.tex_names[key]) + # Expressions + for ename, expr in self.parent.exprs.items(): + self.inputs_dict[ename] = sp.symbols(f'{ename}') + if expr.no_parse: + self.sub_map[rf"\b{ename}\b"] = f"self.rtn.{ename}.v" + else: + self.sub_map[rf"\b{ename}\b"] = f"self.om.{ename}" + self.tex_map[rf"\b{ename}\b"] = f'{expr.tex_name}' + self.val_map[rf"\b{ename}\b"] = f"rtn.{ename}.v" + # Vars for vname, var in self.parent.vars.items(): - tmp = sp.symbols(f'{var.name}') - # tmp = sp.symbols(var.name) - self.inputs_dict[vname] = tmp + self.inputs_dict[vname] = sp.symbols(f'{vname}') self.sub_map[rf"\b{vname}\b"] = f"self.om.{vname}" self.tex_map[rf"\b{vname}\b"] = rf'{var.tex_name}' self.val_map[rf"\b{vname}\b"] = f"rtn.{vname}.v" @@ -164,10 +172,6 @@ def generate_symbols(self, force_generate=False): if not service.no_parse: self.val_map[rf"\b{sname}\b"] = f"rtn.{sname}.v" - # Expressions - for ename, expr in self.parent.exprs.items(): - self.sub_map[rf"\b{ename}\b"] = f"self.rtn.{ename}.optz" - # Constraints # NOTE: constraints are included in sub_map for ExpressionCalc # thus, they don't have the suffix `.v` diff --git a/ams/opt/expr.py b/ams/opt/expr.py index 63a914df..c390e7e9 100644 --- a/ams/opt/expr.py +++ b/ams/opt/expr.py @@ -32,6 +32,8 @@ class Expression(OptzBase): Expression name. One should typically assigning the name directly because it will be automatically assigned by the model. The value of ``name`` will be the symbol name to be used in expressions. + tex_name : str, optional + LaTeX-formatted variable symbol. Defaults to the value of ``name``. info : str, optional Descriptive information unit : str, optional @@ -52,6 +54,7 @@ class Expression(OptzBase): def __init__(self, name: Optional[str] = None, + tex_name: Optional[str] = None, info: Optional[str] = None, unit: Optional[str] = None, e_str: Optional[str] = None, @@ -62,6 +65,7 @@ def __init__(self, horizon: Optional[str] = None, ): OptzBase.__init__(self, name=name, info=info, unit=unit) + self.tex_name = tex_name self.e_str = e_str self.optz = None self.code = None @@ -127,7 +131,7 @@ def evaluate(self): msg = f" - Expression <{self.name}>: {self.code}" logger.debug(pretty_long_message(msg, _prefix, max_length=_max_length)) try: - local_vars = {'self': self, 'np': np, 'cp': cp} + local_vars = {'self': self, 'np': np, 'cp': cp, 'sub_map': self.om.rtn.syms.val_map} self.optz = self._evaluate_expression(self.code, local_vars=local_vars) except Exception as e: raise Exception(f"Error in evaluating Expression <{self.name}>.\n{e}") diff --git a/ams/opt/omodel.py b/ams/opt/omodel.py index 8e92005c..cda85569 100644 --- a/ams/opt/omodel.py +++ b/ams/opt/omodel.py @@ -187,6 +187,7 @@ def _evaluate_exprs(self): for key, val in self.rtn.exprs.items(): try: val.evaluate() + setattr(self, key, val.optz) except Exception as e: raise Exception(f"Failed to evaluate Expression <{key}>.\n{e}") @@ -225,11 +226,12 @@ def evaluate(self, force=False): logger.warning(f"Evaluating OModel for <{self.rtn.class_name}>") t, _ = elapsed() + # NOTE: should evaluate in sequence self._evaluate_params() self._evaluate_vars() + self._evaluate_exprs() self._evaluate_constrs() self._evaluate_obj() - self._evaluate_exprs() self._evaluate_exprcs() self.evaluated = True From 6de04ae9bc7e0e6fd1dc114a4e488bd267b2792e Mon Sep 17 00:00:00 2001 From: jinningwang Date: Sat, 16 Nov 2024 07:07:52 -0500 Subject: [PATCH 33/96] Refactor DCOPF line flow constraints using Expression --- ams/routines/dcopf.py | 20 +++++++++++++------- 1 file changed, 13 insertions(+), 7 deletions(-) diff --git a/ams/routines/dcopf.py b/ams/routines/dcopf.py index 66d5eb00..e4461fea 100644 --- a/ams/routines/dcopf.py +++ b/ams/routines/dcopf.py @@ -9,7 +9,7 @@ from ams.routines.routine import RoutineBase -from ams.opt import Var, Constraint, Objective, ExpressionCalc +from ams.opt import Var, Constraint, Objective, ExpressionCalc, Expression logger = logging.getLogger(__name__) @@ -191,22 +191,23 @@ def __init__(self, system, config): self.pb = Constraint(name='pb', info='power balance', e_str=pb, is_eq=True,) # --- line flow --- + self.plf = Expression(info='Line flow', + name='plf', tex_name=r'p_{lf}', + unit='p.u.', + e_str='Bf@aBus + Pfinj', + model='Line', src=None,) self.plflb = Constraint(info='line flow lower bound', name='plflb', is_eq=False, - e_str='-Bf@aBus - Pfinj - mul(ul, rate_a)',) + e_str='-plf - mul(ul, rate_a)',) self.plfub = Constraint(info='line flow upper bound', name='plfub', is_eq=False, - e_str='Bf@aBus + Pfinj - mul(ul, rate_a)',) + e_str='plf - mul(ul, rate_a)',) self.alflb = Constraint(info='line angle difference lower bound', name='alflb', is_eq=False, e_str='-CftT@aBus + amin',) self.alfub = Constraint(info='line angle difference upper bound', name='alfub', is_eq=False, e_str='CftT@aBus - amax',) - self.plf = ExpressionCalc(info='Line flow', - name='plf', unit='p.u.', - e_str='Bf@aBus + Pfinj', - model='Line', src=None,) # NOTE: in CVXPY, dual_variables returns a list self.pi = ExpressionCalc(info='LMP, dual of ', name='pi', unit='$/p.u.', @@ -279,6 +280,11 @@ def _post_solve(self): """ Post-solve calculations. """ + # NOTE: unpack Expressions if owner and arc are available + for expr in self.exprs.values(): + if expr.owner and expr.src: + expr.owner.set(src=expr.src, attr='v', + idx=expr.get_idx(), value=expr.v) return True def unpack(self, **kwargs): From 3dbfd4ff0385fcccb6d3a959554faa046483e3ac Mon Sep 17 00:00:00 2001 From: jinningwang Date: Sat, 16 Nov 2024 07:12:43 -0500 Subject: [PATCH 34/96] Include class Expression in documenter --- ams/core/documenter.py | 54 +++++++++++++++++++++++++++++++++++++++++- 1 file changed, 53 insertions(+), 1 deletion(-) diff --git a/ams/core/documenter.py b/ams/core/documenter.py index 0cf51a22..22bab272 100644 --- a/ams/core/documenter.py +++ b/ams/core/documenter.py @@ -223,9 +223,10 @@ def get(self, max_width=78, export='plain'): # add tables self.parent.syms.generate_symbols() out += self._obj_doc(max_width=max_width, export=export) + out += self._expr_doc(max_width=max_width, export=export) out += self._constr_doc(max_width=max_width, export=export) - out += self._exprc_doc(max_width=max_width, export=export) out += self._var_doc(max_width=max_width, export=export) + out += self._exprc_doc(max_width=max_width, export=export) out += self._service_doc(max_width=max_width, export=export) out += self._param_doc(max_width=max_width, export=export) out += self.config.doc(max_width=max_width, export=export) @@ -315,6 +316,57 @@ def _constr_doc(self, max_width=78, export='plain'): plain_dict=plain_dict, rest_dict=rest_dict) + def _expr_doc(self, max_width=78, export='plain'): + # Expression documentation + if len(self.parent.exprs) == 0: + return '' + + # prepare temporary lists + names, info = list(), list() + units, sources, units_rest = list(), list(), list() + + for p in self.parent.exprs.values(): + names.append(p.name) + info.append(p.info if p.info else '') + units.append(p.unit if p.unit else '') + units_rest.append(f'*{p.unit}*' if p.unit else '') + + slist = [] + if p.owner is not None and p.src is not None: + slist.append(f'{p.owner.class_name}.{p.src}') + sources.append(','.join(slist)) + + # expressions based on output format + expressions = [] + if export == 'rest': + for p in self.parent.exprs.values(): + expr = _tex_pre(self, p, self.parent.syms.tex_map) + logger.debug(f'{p.name} math: {expr}') + expressions.append(expr) + + title = 'Expressions\n----------------------------------' + else: + title = 'Expressions' + expressions = math_wrap(expressions, export=export) + + plain_dict = OrderedDict([('Name', names), + ('Description', info), + ('Unit', units), + ]) + rest_dict = OrderedDict([('Name', names), + ('Description', info), + ('Expression', expressions), + ('Unit', units_rest), + ('Source', sources), + ]) + + # convert to rows and export as table + return make_doc_table(title=title, + max_width=max_width, + export=export, + plain_dict=plain_dict, + rest_dict=rest_dict) + def _exprc_doc(self, max_width=78, export='plain'): # ExpressionCalc documentation if len(self.parent.exprcs) == 0: From 7471a0eb7cbc425a1bae6bb3742c6c984a231327 Mon Sep 17 00:00:00 2001 From: jinningwang Date: Sat, 16 Nov 2024 07:35:02 -0500 Subject: [PATCH 35/96] Refactor DCOPF generator power limits using Expression --- ams/routines/dcopf.py | 14 ++++++++++---- ams/routines/rted.py | 4 ++-- 2 files changed, 12 insertions(+), 6 deletions(-) diff --git a/ams/routines/dcopf.py b/ams/routines/dcopf.py index e4461fea..55b7a495 100644 --- a/ams/routines/dcopf.py +++ b/ams/routines/dcopf.py @@ -168,12 +168,18 @@ def __init__(self, system, config): name='pg', tex_name=r'p_g', model='StaticGen', src='p', v0=self.pg0) - pglb = '-pg + mul(nctrle, pg0) + mul(ctrle, pmin)' + self.pmaxe = Expression(info='Effective Gen maximum active power', + name='pmaxe', tex_name=r'p_{g, max, e}', + e_str='mul(nctrle, pg0) + mul(ctrle, pmax)', + model='StaticGen', src=None,) + self.pmine = Expression(info='Effective Gen minimum active power', + name='pmine', tex_name=r'p_{g, min, e}', + e_str='mul(nctrle, pg0) + mul(ctrle, pmin)', + model='StaticGen', src=None,) self.pglb = Constraint(name='pglb', info='pg min', - e_str=pglb, is_eq=False,) - pgub = 'pg - mul(nctrle, pg0) - mul(ctrle, pmax)' + e_str='-pg + pmine', is_eq=False,) self.pgub = Constraint(name='pgub', info='pg max', - e_str=pgub, is_eq=False,) + e_str='pg - pmaxe', is_eq=False,) # --- bus --- self.aBus = Var(info='Bus voltage angle', unit='rad', diff --git a/ams/routines/rted.py b/ams/routines/rted.py index 28099dc1..36ecccf8 100644 --- a/ams/routines/rted.py +++ b/ams/routines/rted.py @@ -161,8 +161,8 @@ def __init__(self, system, config): self.rbu.e_str = 'gs @ mul(ug, pru) - dud' self.rbd.e_str = 'gs @ mul(ug, prd) - ddd' # RegUp/Dn reserve source - self.rru.e_str = 'mul(ug, (pg + pru)) - mul(ug, pmax)' - self.rrd.e_str = 'mul(ug, (-pg + prd)) + mul(ug, pmin)' + self.rru.e_str = 'mul(ug, (pg + pru)) - mul(ug, pmaxe)' + self.rrd.e_str = 'mul(ug, (-pg + prd)) + mul(ug, pmine)' # Gen ramping up/down self.rgu.e_str = 'mul(ug, (pg-pg0-R10))' self.rgd.e_str = 'mul(ug, (-pg+pg0-R10))' From a6e099009d9f0391d1ee6db51af0ef01f4347da7 Mon Sep 17 00:00:00 2001 From: jinningwang Date: Sat, 16 Nov 2024 07:36:58 -0500 Subject: [PATCH 36/96] Typo --- ams/routines/dcopf.py | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/ams/routines/dcopf.py b/ams/routines/dcopf.py index 55b7a495..11644fe8 100644 --- a/ams/routines/dcopf.py +++ b/ams/routines/dcopf.py @@ -168,11 +168,11 @@ def __init__(self, system, config): name='pg', tex_name=r'p_g', model='StaticGen', src='p', v0=self.pg0) - self.pmaxe = Expression(info='Effective Gen maximum active power', + self.pmaxe = Expression(info='Effective pmax', name='pmaxe', tex_name=r'p_{g, max, e}', e_str='mul(nctrle, pg0) + mul(ctrle, pmax)', model='StaticGen', src=None,) - self.pmine = Expression(info='Effective Gen minimum active power', + self.pmine = Expression(info='Effective pmin', name='pmine', tex_name=r'p_{g, min, e}', e_str='mul(nctrle, pg0) + mul(ctrle, pmin)', model='StaticGen', src=None,) From 09bc28bff16677b359c4bed938c3718ce5776768 Mon Sep 17 00:00:00 2001 From: jinningwang Date: Sat, 16 Nov 2024 07:44:24 -0500 Subject: [PATCH 37/96] Refactor ED pg limits using Expression --- ams/routines/ed.py | 12 ++++++------ 1 file changed, 6 insertions(+), 6 deletions(-) diff --git a/ams/routines/ed.py b/ams/routines/ed.py index f5624d73..944408e4 100644 --- a/ams/routines/ed.py +++ b/ams/routines/ed.py @@ -158,12 +158,12 @@ def __init__(self, system, config): # --- gen --- self.ctrle.u2 = self.ugt self.nctrle.u2 = self.ugt - pglb = '-pg + mul(mul(nctrle, pg0), tlv) ' - pglb += '+ mul(mul(ctrle, tlv), pmin)' - self.pglb.e_str = pglb - pgub = 'pg - mul(mul(nctrle, pg0), tlv) ' - pgub += '- mul(mul(ctrle, tlv), pmax)' - self.pgub.e_str = pgub + pmaxe = 'mul(mul(nctrle, pg0), tlv) + mul(mul(ctrle, tlv), pmax)' + self.pmaxe.e_str = pmaxe + pmine = 'mul(mul(nctrle, pg0), tlv) + mul(mul(ctrle, tlv), pmin)' + self.pmine.e_str = pmine + self.pglb.e_str = '-pg + pmine' + self.pgub.e_str = 'pg - pmaxe' self.pru.horizon = self.timeslot self.pru.info = '2D RegUp power' From 761ad85d1915593cab9dd70b19f939142b7f049b Mon Sep 17 00:00:00 2001 From: jinningwang Date: Sat, 16 Nov 2024 07:51:01 -0500 Subject: [PATCH 38/96] Refactor UC pg limits using Expression --- ams/routines/uc.py | 12 ++++++------ 1 file changed, 6 insertions(+), 6 deletions(-) diff --git a/ams/routines/uc.py b/ams/routines/uc.py index 92527de8..f3f49df3 100644 --- a/ams/routines/uc.py +++ b/ams/routines/uc.py @@ -153,12 +153,12 @@ def __init__(self, system, config): self.ctrle.info = 'Reshaped controllability' self.nctrle.u2 = self.tlv self.nctrle.info = 'Reshaped non-controllability' - pglb = '-pg + mul(mul(nctrl, pg0), ugd)' - pglb += '+ mul(mul(ctrl, pmin), ugd)' - self.pglb.e_str = pglb - pgub = 'pg - mul(mul(nctrl, pg0), ugd)' - pgub += '- mul(mul(ctrl, pmax), ugd)' - self.pgub.e_str = pgub + pmaxe = 'mul(mul(nctrl, pg0), ugd) + mul(mul(ctrl, pmax), ugd)' + self.pmaxe.e_str = pmaxe + pmine = 'mul(mul(ctrl, pmin), ugd) + mul(mul(nctrl, pg0), ugd)' + self.pmine.e_str = pmine + self.pglb.e_str = '-pg + pmine' + self.pgub.e_str = 'pg - pmaxe' self.ugd = Var(info='commitment decision', horizon=self.timeslot, From 8034bb2b575216f3b9e9a1f1e6292ded984ce31e Mon Sep 17 00:00:00 2001 From: jinningwang Date: Sat, 16 Nov 2024 08:54:50 -0500 Subject: [PATCH 39/96] Use in-place assign for class Expression._v --- ams/opt/expr.py | 2 +- ams/routines/pflow2.py | 147 ++++++++++------------------------------- 2 files changed, 37 insertions(+), 112 deletions(-) diff --git a/ams/opt/expr.py b/ams/opt/expr.py index c390e7e9..51532bc6 100644 --- a/ams/opt/expr.py +++ b/ams/opt/expr.py @@ -171,7 +171,7 @@ def v(self, value): Set the value. """ if self.no_parse: - self._v = value + self._v[...] = value else: logger.warning('Cannot set value to an Expression that is not no_parse.') diff --git a/ams/routines/pflow2.py b/ams/routines/pflow2.py index 19122363..f68b4780 100644 --- a/ams/routines/pflow2.py +++ b/ams/routines/pflow2.py @@ -204,36 +204,43 @@ def __init__(self, system, config): name='sbus', is_eq=True, e_str='csb@aBus',) # --- power balance --- - self.Vc = Expression(info='Bus voltage in complex', - name='Vc', unit='p.u.', + # NOTE: CVXPY does not support exp complex number + self.Vc = Expression(info='Complex bus voltage', + name='Vc', tex_name=r'V_{bus}', + unit='p.u.', no_parse=True, e_str='vBus dot exp(1j dot aBus)', - no_parse=True, model='Bus', src=None, vtype=complex,) - # Vc = vBus * exp(1j * aBus) - # TODO: pout = VYV; diag(Vc) @ conj(Y) @ Vc - # TODO: pin = $injected power$ - # TODO: pb = pout - pin + self.pbin = Expression(info='Bus power in', + name='pbin', tex_name=r'p_{bus}^{in}', + unit='p.u.', no_parse=True, + e_str='-Cl@pd - Csh@gsh + Cg@pg', + model='Bus', src=None,) + self.pbout = Expression(info='Bus power out', + name='pbout', tex_name=r'p_{bus}^{out}', + unit='p.u.', no_parse=True, + e_str='Vc @ Bbus @ conj(Vc)', + model='Bus', src=None) pb = 'Bbus@aBus + Pbusinj + Cl@pd + Csh@gsh - Cg@pg' self.pb = Constraint(name='pb', info='power balance', - e_str=pb, is_eq=True,) - # --- line flow --- - self.plf = Var(info='Line flow', - unit='p.u.', - name='plf', tex_name=r'p_{lf}', - model='Line',) - self.plflb = Constraint(info='line flow lower bound', - name='plflb', is_eq=False, - e_str='-Bf@aBus - Pfinj - mul(ul, rate_a)',) - self.plfub = Constraint(info='line flow upper bound', - name='plfub', is_eq=False, - e_str='Bf@aBus + Pfinj - mul(ul, rate_a)',) - self.alflb = Constraint(info='line angle difference lower bound', - name='alflb', is_eq=False, - e_str='-CftT@aBus + amin',) - self.alfub = Constraint(info='line angle difference upper bound', - name='alfub', is_eq=False, - e_str='CftT@aBus - amax',) + e_str='pbin - pbout', is_eq=True,) + # # --- line flow --- + # self.plf = Var(info='Line flow', + # unit='p.u.', + # name='plf', tex_name=r'p_{lf}', + # model='Line',) + # self.plflb = Constraint(info='line flow lower bound', + # name='plflb', is_eq=False, + # e_str='-Bf@aBus - Pfinj - mul(ul, rate_a)',) + # self.plfub = Constraint(info='line flow upper bound', + # name='plfub', is_eq=False, + # e_str='Bf@aBus + Pfinj - mul(ul, rate_a)',) + # self.alflb = Constraint(info='line angle difference lower bound', + # name='alflb', is_eq=False, + # e_str='-CftT@aBus + amin',) + # self.alfub = Constraint(info='line angle difference upper bound', + # name='alfub', is_eq=False, + # e_str='CftT@aBus - amax',) # --- objective --- self.obj = Objective(name='obj', @@ -245,7 +252,8 @@ def solve(self, **kwargs): Solve the routine optimization model. args and kwargs go to `self.om.prob.solve()` (`cvxpy.Problem.solve()`). """ - return self.om.prob.solve(**kwargs) + raise NotImplementedError('Not implemented yet.') + def run(self, **kwargs): """ @@ -292,99 +300,16 @@ def run(self, **kwargs): method : function, optional A custom solve method to use. """ - return super().run(**kwargs) + raise NotImplementedError('Not implemented yet.') def _post_solve(self): """ Post-solve calculations. """ - for expr in self.exprs.values(): - try: - var = getattr(self, expr.var) - var.optz.value = expr.v - logger.debug(f'Post solve: {var} = {expr.e_str}') - except AttributeError: - raise AttributeError(f'No such variable {expr.var}') return True def unpack(self, **kwargs): """ Unpack the results from CVXPY model. """ - # --- solver results to routine algeb --- - for _, var in self.vars.items(): - # --- copy results from routine algeb into system algeb --- - if var.model is None: # if no owner - continue - if var.src is None: # if no source - continue - else: - try: - idx = var.owner.get_idx() - except AttributeError: - idx = var.owner.idx.v - else: - pass - # NOTE: only unpack the variables that are in the model or group - try: - var.owner.set(src=var.src, idx=idx, attr='v', value=var.v) - except (KeyError, TypeError): - logger.error(f'Failed to unpack <{var}> to <{var.owner.class_name}>.') - pass - - # label the most recent solved routine - self.system.recent = self.system.routines[self.class_name] - return True - - def dc2ac(self, kloss=1.0, **kwargs): - """ - Convert the DCOPF results with ACOPF. - - Parameters - ---------- - kloss : float, optional - The loss factor for the conversion. Defaults to 1.2. - """ - exec_time = self.exec_time - if self.exec_time == 0 or self.exit_code != 0: - logger.warning(f'{self.class_name} is not executed successfully, quit conversion.') - return False - - # --- ACOPF --- - # scale up load - pq_idx = self.system.StaticLoad.get_idx() - pd0 = self.system.StaticLoad.get(src='p0', attr='v', idx=pq_idx).copy() - qd0 = self.system.StaticLoad.get(src='q0', attr='v', idx=pq_idx).copy() - self.system.StaticLoad.set(src='p0', idx=pq_idx, attr='v', value=pd0 * kloss) - self.system.StaticLoad.set(src='q0', idx=pq_idx, attr='v', value=qd0 * kloss) - - ACOPF = self.system.ACOPF - # run ACOPF - ACOPF.run() - # self.exec_time += ACOPF.exec_time - # scale load back - self.system.StaticLoad.set(src='p0', idx=pq_idx, value=pd0) - self.system.StaticLoad.set(src='q0', idx=pq_idx, value=qd0) - if not ACOPF.exit_code == 0: - logger.warning(' did not converge, conversion failed.') - # NOTE: mock results to fit interface with ANDES - self.vBus = ACOPF.vBus - self.vBus.optz.value = np.ones(self.system.Bus.n) - self.aBus.optz.value = np.zeros(self.system.Bus.n) - return False - self.pg.optz.value = ACOPF.pg.v - - # NOTE: mock results to fit interface with ANDES - self.addVars(name='vBus', - info='Bus voltage', unit='p.u.', - model='Bus', src='v',) - self.vBus.parse() - self.vBus.optz.value = ACOPF.vBus.v - self.aBus.optz.value = ACOPF.aBus.v - self.exec_time = exec_time - - # --- set status --- - self.system.recent = self - self.converted = True - logger.warning(f'<{self.class_name}> converted to AC.') - return True + raise NotImplementedError('Not implemented yet.') \ No newline at end of file From 2987977f82caeac8adc6a57660e8f89328634f21 Mon Sep 17 00:00:00 2001 From: jinningwang Date: Sat, 16 Nov 2024 08:57:29 -0500 Subject: [PATCH 40/96] Rename file expr to expression for calrity --- ams/opt/__init__.py | 2 +- ams/opt/{expr.py => expression.py} | 0 2 files changed, 1 insertion(+), 1 deletion(-) rename ams/opt/{expr.py => expression.py} (100%) diff --git a/ams/opt/__init__.py b/ams/opt/__init__.py index c468e92f..9cc99133 100644 --- a/ams/opt/__init__.py +++ b/ams/opt/__init__.py @@ -8,5 +8,5 @@ from ams.opt.param import Param # NOQA from ams.opt.constraint import Constraint # NOQA from ams.opt.objective import Objective # NOQA -from ams.opt.expr import Expression # NOQA +from ams.opt.expression import Expression # NOQA from ams.opt.omodel import OModel # NOQA diff --git a/ams/opt/expr.py b/ams/opt/expression.py similarity index 100% rename from ams/opt/expr.py rename to ams/opt/expression.py From a8b9a98c37e9c49a8c537d069478c19577ecf2fd Mon Sep 17 00:00:00 2001 From: jinningwang Date: Sat, 16 Nov 2024 10:00:17 -0500 Subject: [PATCH 41/96] Fix symprocessor sub_map and val_map for Expression when no_parse --- ams/core/symprocessor.py | 21 +++++++++++---------- 1 file changed, 11 insertions(+), 10 deletions(-) diff --git a/ams/core/symprocessor.py b/ams/core/symprocessor.py index cde68fe0..1cd09167 100644 --- a/ams/core/symprocessor.py +++ b/ams/core/symprocessor.py @@ -90,6 +90,8 @@ def __init__(self, parent): (r'cp\.(Minimize|Maximize)', r'float'), # remove cp.Minimize/Maximize (r'\bcp.\b', 'np.'), (r'\bexp\b', 'np.exp'), + (r'\blog\b', 'np.log'), + (r'\bconj\b', 'np.conj'), ]) self.status = { @@ -123,16 +125,6 @@ def generate_symbols(self, force_generate=False): for key in self.parent.tex_names.keys(): self.tex_names[key] = sp.symbols(self.parent.tex_names[key]) - # Expressions - for ename, expr in self.parent.exprs.items(): - self.inputs_dict[ename] = sp.symbols(f'{ename}') - if expr.no_parse: - self.sub_map[rf"\b{ename}\b"] = f"self.rtn.{ename}.v" - else: - self.sub_map[rf"\b{ename}\b"] = f"self.om.{ename}" - self.tex_map[rf"\b{ename}\b"] = f'{expr.tex_name}' - self.val_map[rf"\b{ename}\b"] = f"rtn.{ename}.v" - # Vars for vname, var in self.parent.vars.items(): self.inputs_dict[vname] = sp.symbols(f'{vname}') @@ -172,6 +164,15 @@ def generate_symbols(self, force_generate=False): if not service.no_parse: self.val_map[rf"\b{sname}\b"] = f"rtn.{sname}.v" + # Expressions + for ename, expr in self.parent.exprs.items(): + self.inputs_dict[ename] = sp.symbols(f'{ename}') + if expr.no_parse: + self.sub_map[rf"\b{ename}\b"] = f"self.rtn.{ename}._v" + else: + self.sub_map[rf"\b{ename}\b"] = f"self.om.{ename}" + self.tex_map[rf"\b{ename}\b"] = f'{expr.tex_name}' + # Constraints # NOTE: constraints are included in sub_map for ExpressionCalc # thus, they don't have the suffix `.v` From 54df0599e8312e9668180c03174021f54eb5cee2 Mon Sep 17 00:00:00 2001 From: jinningwang Date: Sat, 16 Nov 2024 16:04:51 -0500 Subject: [PATCH 42/96] [WIP] PFlow2, formulation --- ams/routines/pflow2.py | 115 ++++------------------------------------- 1 file changed, 10 insertions(+), 105 deletions(-) diff --git a/ams/routines/pflow2.py b/ams/routines/pflow2.py index f68b4780..21f09020 100644 --- a/ams/routines/pflow2.py +++ b/ams/routines/pflow2.py @@ -3,13 +3,13 @@ """ import logging -import numpy as np +import numpy as np # noqa from ams.core.param import RParam -from ams.core.service import NumOp, NumOpDual, VarSelect +from ams.core.service import NumOp, NumOpDual, VarSelect # noqa from ams.routines.routine import RoutineBase -from ams.opt import Var, Constraint, Objective, Expression +from ams.opt import Var, Constraint, Objective, Expression # noqa logger = logging.getLogger(__name__) @@ -52,46 +52,6 @@ def __init__(self, system, config): name='ug', tex_name=r'u_{g}', model='StaticGen', src='u', no_parse=True) - self.ctrl = RParam(info='Gen controllability', - name='ctrl', tex_name=r'c_{trl}', - model='StaticGen', src='ctrl', - no_parse=True) - self.ctrle = NumOpDual(info='Effective Gen controllability', - name='ctrle', tex_name=r'c_{trl, e}', - u=self.ctrl, u2=self.ug, - fun=np.multiply, no_parse=True) - self.nctrl = NumOp(u=self.ctrl, fun=np.logical_not, - name='nctrl', tex_name=r'c_{trl,n}', - info='Effective Gen uncontrollability', - no_parse=True,) - self.nctrle = NumOpDual(info='Effective Gen uncontrollability', - name='nctrle', tex_name=r'c_{trl,n,e}', - u=self.nctrl, u2=self.ug, - fun=np.multiply, no_parse=True) - self.pmax = RParam(info='Gen maximum active power', - name='pmax', tex_name=r'p_{g, max}', - unit='p.u.', model='StaticGen', - no_parse=False,) - self.pmin = RParam(info='Gen minimum active power', - name='pmin', tex_name=r'p_{g, min}', - unit='p.u.', model='StaticGen', - no_parse=False,) - self.pg0 = RParam(info='Gen initial active power', - name='pg0', tex_name=r'p_{g, 0}', - unit='p.u.', - model='StaticGen', src='p0') - self.qmax = RParam(info='Gen maximum reactive power', - name='qmax', tex_name=r'q_{g, max}', - unit='p.u.', model='StaticGen', - no_parse=False,) - self.qmin = RParam(info='Gen minimum reactive power', - name='qmin', tex_name=r'q_{g, min}', - unit='p.u.', model='StaticGen', - no_parse=False,) - self.qg0 = RParam(info='Gen initial reactive power', - name='qg0', tex_name=r'q_{g, 0}', - unit='p.u.', - model='StaticGen', src='q0') # --- bus --- self.buss = RParam(info='Bus slack', name='buss', tex_name=r'B_{us,s}', @@ -168,25 +128,11 @@ def __init__(self, system, config): self.pg = Var(info='Gen active power', unit='p.u.', name='pg', tex_name=r'p_g', - model='StaticGen', src='p', - v0=self.pg0) - pglb = '-pg + mul(nctrle, pg0) + mul(ctrle, pmin)' - self.pglb = Constraint(name='pglb', info='pg min', - e_str=pglb, is_eq=False,) - pgub = 'pg - mul(nctrle, pg0) - mul(ctrle, pmax)' - self.pgub = Constraint(name='pgub', info='pg max', - e_str=pgub, is_eq=False,) + model='StaticGen', src='p',) self.qg = Var(info='Gen reactive power', unit='p.u.', name='qg', tex_name=r'q_g', model='StaticGen', src='q',) - qglb = '-qg + mul(nctrle, qg0) + mul(ctrle, qmin)' - self.qglb = Constraint(name='qglb', info='qg min', - e_str=qglb, is_eq=False,) - qgub = 'qg - mul(nctrle, qg0) - mul(ctrle, qmax)' - self.qgub = Constraint(name='qgub', info='qg max', - e_str=qgub, is_eq=False,) - # --- bus --- self.aBus = Var(info='Bus voltage angle', unit='rad', @@ -196,56 +142,16 @@ def __init__(self, system, config): unit='p.u.', name='vBus', tex_name=r'v_{bus}', model='Bus', src='v',) - self.csb = VarSelect(info='select slack bus', - name='csb', tex_name=r'c_{sb}', - u=self.aBus, indexer='buss', - no_parse=True,) - self.sba = Constraint(info='align slack bus angle', - name='sbus', is_eq=True, - e_str='csb@aBus',) # --- power balance --- - # NOTE: CVXPY does not support exp complex number - self.Vc = Expression(info='Complex bus voltage', - name='Vc', tex_name=r'V_{bus}', - unit='p.u.', no_parse=True, - e_str='vBus dot exp(1j dot aBus)', - model='Bus', src=None, - vtype=complex,) - self.pbin = Expression(info='Bus power in', - name='pbin', tex_name=r'p_{bus}^{in}', - unit='p.u.', no_parse=True, - e_str='-Cl@pd - Csh@gsh + Cg@pg', - model='Bus', src=None,) - self.pbout = Expression(info='Bus power out', - name='pbout', tex_name=r'p_{bus}^{out}', - unit='p.u.', no_parse=True, - e_str='Vc @ Bbus @ conj(Vc)', - model='Bus', src=None) - pb = 'Bbus@aBus + Pbusinj + Cl@pd + Csh@gsh - Cg@pg' - self.pb = Constraint(name='pb', info='power balance', - e_str='pbin - pbout', is_eq=True,) - # # --- line flow --- - # self.plf = Var(info='Line flow', - # unit='p.u.', - # name='plf', tex_name=r'p_{lf}', - # model='Line',) - # self.plflb = Constraint(info='line flow lower bound', - # name='plflb', is_eq=False, - # e_str='-Bf@aBus - Pfinj - mul(ul, rate_a)',) - # self.plfub = Constraint(info='line flow upper bound', - # name='plfub', is_eq=False, - # e_str='Bf@aBus + Pfinj - mul(ul, rate_a)',) - # self.alflb = Constraint(info='line angle difference lower bound', - # name='alflb', is_eq=False, - # e_str='-CftT@aBus + amin',) - # self.alfub = Constraint(info='line angle difference upper bound', - # name='alfub', is_eq=False, - # e_str='CftT@aBus - amax',) + # pb = 'Bbus@aBus + Pbusinj + Cl@pd + Csh@gsh - Cg@pg' + # self.pb = Constraint(name='pb', info='power balance', + # e_str='pbin - pbout', is_eq=True, + # ) # --- objective --- self.obj = Objective(name='obj', info='No objective', unit='$', - sense='min', e_str='0',) + sense='min', e_str='(0)',) def solve(self, **kwargs): """ @@ -253,7 +159,6 @@ def solve(self, **kwargs): args and kwargs go to `self.om.prob.solve()` (`cvxpy.Problem.solve()`). """ raise NotImplementedError('Not implemented yet.') - def run(self, **kwargs): """ @@ -312,4 +217,4 @@ def unpack(self, **kwargs): """ Unpack the results from CVXPY model. """ - raise NotImplementedError('Not implemented yet.') \ No newline at end of file + raise NotImplementedError('Not implemented yet.') From 4279dfd555009b3a8ac88ae6d3806ba373b8d38c Mon Sep 17 00:00:00 2001 From: jinningwang Date: Sat, 16 Nov 2024 16:26:15 -0500 Subject: [PATCH 43/96] [WIP] PFlow2, formulation --- ams/routines/pflow2.py | 31 +++++++++++-------------------- 1 file changed, 11 insertions(+), 20 deletions(-) diff --git a/ams/routines/pflow2.py b/ams/routines/pflow2.py index 21f09020..64777f15 100644 --- a/ams/routines/pflow2.py +++ b/ams/routines/pflow2.py @@ -52,11 +52,6 @@ def __init__(self, system, config): name='ug', tex_name=r'u_{g}', model='StaticGen', src='u', no_parse=True) - # --- bus --- - self.buss = RParam(info='Bus slack', - name='buss', tex_name=r'B_{us,s}', - model='Slack', src='bus', - no_parse=True,) # --- load --- self.pd = RParam(info='active demand', name='pd', tex_name=r'p_{d}', @@ -71,18 +66,6 @@ def __init__(self, system, config): name='ul', tex_name=r'u_{l}', model='Line', src='u', no_parse=True,) - self.rate_a = RParam(info='long-term flow limit', - name='rate_a', tex_name=r'R_{ATEA}', - unit='p.u.', model='Line',) - # --- line angle difference --- - self.amax = RParam(model='Line', src='amax', - name='amax', tex_name=r'\theta_{bus, max}', - info='max line angle difference', - no_parse=True,) - self.amin = RParam(model='Line', src='amin', - name='amin', tex_name=r'\theta_{bus, min}', - info='min line angle difference', - no_parse=True,) # --- shunt --- self.gsh = RParam(info='shunt conductance', name='gsh', tex_name=r'g_{sh}', @@ -149,9 +132,17 @@ def __init__(self, system, config): # ) # --- objective --- - self.obj = Objective(name='obj', - info='No objective', unit='$', - sense='min', e_str='(0)',) + # self.obj = Objective(name='obj', + # info='No objective', unit='$', + # sense='min', e_str='(0)',) + + # TODO: seems we might need override the following methods + def init(self, **kwargs): + """ + Initialize the routine optimization model. + args and kwargs go to `self.om.init()` (`cvxpy.Problem.__init__()`). + """ + raise NotImplementedError('Not implemented yet.') def solve(self, **kwargs): """ From 3b6991f0ad11258fc69ea120f0c52acea8f04206 Mon Sep 17 00:00:00 2001 From: jinningwang Date: Sat, 16 Nov 2024 17:14:40 -0500 Subject: [PATCH 44/96] [WIP] PFlow2, formulation --- ams/routines/pflow2.py | 17 +++++++++++------ 1 file changed, 11 insertions(+), 6 deletions(-) diff --git a/ams/routines/pflow2.py b/ams/routines/pflow2.py index 64777f15..e902b416 100644 --- a/ams/routines/pflow2.py +++ b/ams/routines/pflow2.py @@ -9,7 +9,7 @@ from ams.routines.routine import RoutineBase -from ams.opt import Var, Constraint, Objective, Expression # noqa +from ams.opt import Var, Constraint, Objective, Expression, ExpressionCalc # noqa logger = logging.getLogger(__name__) @@ -126,16 +126,21 @@ def __init__(self, system, config): name='vBus', tex_name=r'v_{bus}', model='Bus', src='v',) # --- power balance --- - # pb = 'Bbus@aBus + Pbusinj + Cl@pd + Csh@gsh - Cg@pg' - # self.pb = Constraint(name='pb', info='power balance', - # e_str='pbin - pbout', is_eq=True, - # ) - + pb = 'Bbus@aBus + Pbusinj + Cl@pd + Csh@gsh - Cg@pg' + self.pb = Constraint(name='pb', info='power balance', + e_str=pb, is_eq=True, + ) + self.Vc = ExpressionCalc(name='Vc', + info='power balance', + e_str='vBus dot exp(1j * aBus)', + ) # --- objective --- # self.obj = Objective(name='obj', # info='No objective', unit='$', # sense='min', e_str='(0)',) + # TODO: we might also need to override self.om.init()? + # TODO: seems we might need override the following methods def init(self, **kwargs): """ From 17b4c063a0146bf7e36a86600f9aa0fbe52789c7 Mon Sep 17 00:00:00 2001 From: jinningwang Date: Sat, 16 Nov 2024 17:43:51 -0500 Subject: [PATCH 45/96] Add a module nlopt for non-linear problems --- ams/nlopt/__init__.py | 5 ++++ ams/nlopt/pfopt.py | 59 +++++++++++++++++++++++++++++++++++++++++++ ams/powerflow.py | 35 +++++++++++++++++++++++++ 3 files changed, 99 insertions(+) create mode 100644 ams/nlopt/__init__.py create mode 100644 ams/nlopt/pfopt.py create mode 100644 ams/powerflow.py diff --git a/ams/nlopt/__init__.py b/ams/nlopt/__init__.py new file mode 100644 index 00000000..69089161 --- /dev/null +++ b/ams/nlopt/__init__.py @@ -0,0 +1,5 @@ +""" +Module for non-linear problems. +""" + +from ams.nlopt.pfopt import PFlowSolver # NOQA diff --git a/ams/nlopt/pfopt.py b/ams/nlopt/pfopt.py new file mode 100644 index 00000000..0f793a97 --- /dev/null +++ b/ams/nlopt/pfopt.py @@ -0,0 +1,59 @@ +""" +Module for non-linear power flow solving. +""" +import logging + +from collections import OrderedDict + + +logger = logging.getLogger(__name__) + + +class PFlowSolver: + """ + Base class for power flow solver. + """ + + def __init__(self, routine): + self.rtn = routine + self.prob = None + self.exprs = OrderedDict() + self.params = OrderedDict() + self.vars = OrderedDict() + self.constrs = OrderedDict() + self.obj = None + self.parsed = False + self.evaluated = False + self.finalized = False + + @property + def initialized(self): + """ + Return the initialization status. + """ + return self.parsed and self.evaluated and self.finalized + + def parse(self, force=False): + raise NotImplementedError + + def evaluate(self, force=False): + raise NotImplementedError + + def finalize(self, force=False): + raise NotImplementedError + + def init(self): + """ + Initialize the power flow solver. + """ + raise NotImplementedError + + def update(self, params): + raise NotImplementedError + + @property + def class_name(self): + """ + Return the class name + """ + return self.__class__.__name__ diff --git a/ams/powerflow.py b/ams/powerflow.py new file mode 100644 index 00000000..141a026a --- /dev/null +++ b/ams/powerflow.py @@ -0,0 +1,35 @@ +""" +Module for non-linear equations. +""" + +from ams.routines.pflow2 import PFlow2 + + +class PFlowSolver: + """ + Base class for power flow solver. + """ + + def __init__(self, rtn: PFlow2): + """ + Initialize the non-linear equations with an instance of PFlow2. + """ + self.rtn = rtn + + def init(self): + """ + Initialize the power flow solver. + """ + raise NotImplementedError + + def parse(self): + """ + Parse the power flow equations. + """ + raise NotImplementedError + + def evaluate(self): + """ + Evaluate the power flow equations. + """ + raise NotImplementedError From b149fa7d2c93c2b436735771bae5af3354a02f5f Mon Sep 17 00:00:00 2001 From: jinningwang Date: Sat, 16 Nov 2024 17:48:58 -0500 Subject: [PATCH 46/96] Remove unused files --- ams/powerflow.py | 35 ----------------------------------- 1 file changed, 35 deletions(-) delete mode 100644 ams/powerflow.py diff --git a/ams/powerflow.py b/ams/powerflow.py deleted file mode 100644 index 141a026a..00000000 --- a/ams/powerflow.py +++ /dev/null @@ -1,35 +0,0 @@ -""" -Module for non-linear equations. -""" - -from ams.routines.pflow2 import PFlow2 - - -class PFlowSolver: - """ - Base class for power flow solver. - """ - - def __init__(self, rtn: PFlow2): - """ - Initialize the non-linear equations with an instance of PFlow2. - """ - self.rtn = rtn - - def init(self): - """ - Initialize the power flow solver. - """ - raise NotImplementedError - - def parse(self): - """ - Parse the power flow equations. - """ - raise NotImplementedError - - def evaluate(self): - """ - Evaluate the power flow equations. - """ - raise NotImplementedError From 16b9ba15eb4a137d4df7c8b6a78930ed116b7539 Mon Sep 17 00:00:00 2001 From: jinningwang Date: Sat, 16 Nov 2024 17:54:35 -0500 Subject: [PATCH 47/96] Rename file pfopt to pfmodel for clarity --- ams/nlopt/__init__.py | 2 +- ams/nlopt/{pfopt.py => pfmodel.py} | 5 ++++- ams/routines/routine.py | 7 ++++++- 3 files changed, 11 insertions(+), 3 deletions(-) rename ams/nlopt/{pfopt.py => pfmodel.py} (90%) diff --git a/ams/nlopt/__init__.py b/ams/nlopt/__init__.py index 69089161..cc850ffe 100644 --- a/ams/nlopt/__init__.py +++ b/ams/nlopt/__init__.py @@ -2,4 +2,4 @@ Module for non-linear problems. """ -from ams.nlopt.pfopt import PFlowSolver # NOQA +from ams.nlopt.pfmodel import PFModel # NOQA diff --git a/ams/nlopt/pfopt.py b/ams/nlopt/pfmodel.py similarity index 90% rename from ams/nlopt/pfopt.py rename to ams/nlopt/pfmodel.py index 0f793a97..49a9d3ac 100644 --- a/ams/nlopt/pfopt.py +++ b/ams/nlopt/pfmodel.py @@ -9,7 +9,7 @@ logger = logging.getLogger(__name__) -class PFlowSolver: +class PFModel: """ Base class for power flow solver. """ @@ -57,3 +57,6 @@ def class_name(self): Return the class name """ return self.__class__.__name__ + + def __repr__(self) -> str: + return f'{self.rtn.class_name}.{self.__class__.__name__} at {hex(id(self))}' diff --git a/ams/routines/routine.py b/ams/routines/routine.py index 2b19d7c5..f4a2cd40 100644 --- a/ams/routines/routine.py +++ b/ams/routines/routine.py @@ -17,6 +17,7 @@ from ams.core.documenter import RDocumenter from ams.core.service import RBaseService, ValueService from ams.opt import OModel, Param, Var, Constraint, Objective, ExpressionCalc, Expression +from ams.nlopt import PFModel from ams.shared import pd @@ -119,7 +120,11 @@ def __init__(self, system=None, config=None): self.map2 = OrderedDict() # to ANDES # --- optimization modeling --- - self.om = OModel(routine=self) # optimization model + # NOTE: for specific routins, the om is a dedicated model + if self.class_name == 'PFlow2': + self.om = PFModel(routine=self) + else: + self.om = OModel(routine=self) # optimization model if config is not None: self.config.load(config) From 9a52c05dc4945c8c3d350d4883ed32c899c46218 Mon Sep 17 00:00:00 2001 From: jinningwang Date: Sat, 16 Nov 2024 18:11:10 -0500 Subject: [PATCH 48/96] [WIP] PFlow2, solver outline --- ams/nlopt/pfmodel.py | 18 ++++++++++++------ ams/routines/pflow2.py | 12 ++++++------ 2 files changed, 18 insertions(+), 12 deletions(-) diff --git a/ams/nlopt/pfmodel.py b/ams/nlopt/pfmodel.py index 49a9d3ac..65736992 100644 --- a/ams/nlopt/pfmodel.py +++ b/ams/nlopt/pfmodel.py @@ -34,22 +34,28 @@ def initialized(self): return self.parsed and self.evaluated and self.finalized def parse(self, force=False): - raise NotImplementedError + self.parsed = True + return self.parsed def evaluate(self, force=False): - raise NotImplementedError + self.evaluated = True + return self.evaluated def finalize(self, force=False): - raise NotImplementedError + self.finalized = True + return self.finalized - def init(self): + def init(self, force=False): """ Initialize the power flow solver. """ - raise NotImplementedError + self.parse(force) + self.evaluate(force) + self.finalize(force) + return self.initialized def update(self, params): - raise NotImplementedError + pass @property def class_name(self): diff --git a/ams/routines/pflow2.py b/ams/routines/pflow2.py index e902b416..475c60f2 100644 --- a/ams/routines/pflow2.py +++ b/ams/routines/pflow2.py @@ -142,12 +142,12 @@ def __init__(self, system, config): # TODO: we might also need to override self.om.init()? # TODO: seems we might need override the following methods - def init(self, **kwargs): - """ - Initialize the routine optimization model. - args and kwargs go to `self.om.init()` (`cvxpy.Problem.__init__()`). - """ - raise NotImplementedError('Not implemented yet.') + # def init(self, **kwargs): + # """ + # Initialize the routine optimization model. + # args and kwargs go to `self.om.init()` (`cvxpy.Problem.__init__()`). + # """ + # raise NotImplementedError('Not implemented yet.') def solve(self, **kwargs): """ From 1899e67376a01ea0727bf9b3322669b39fcd5bcc Mon Sep 17 00:00:00 2001 From: jinningwang Date: Sat, 16 Nov 2024 18:16:50 -0500 Subject: [PATCH 49/96] [WIP] PFlow2, minor change --- ams/nlopt/pfmodel.py | 23 ++++------------------- 1 file changed, 4 insertions(+), 19 deletions(-) diff --git a/ams/nlopt/pfmodel.py b/ams/nlopt/pfmodel.py index 65736992..6ac250f9 100644 --- a/ams/nlopt/pfmodel.py +++ b/ams/nlopt/pfmodel.py @@ -3,36 +3,21 @@ """ import logging -from collections import OrderedDict +from ams.opt import OModel logger = logging.getLogger(__name__) -class PFModel: +class PFModel(OModel): """ Base class for power flow solver. """ def __init__(self, routine): - self.rtn = routine - self.prob = None - self.exprs = OrderedDict() - self.params = OrderedDict() - self.vars = OrderedDict() - self.constrs = OrderedDict() - self.obj = None - self.parsed = False - self.evaluated = False - self.finalized = False - - @property - def initialized(self): - """ - Return the initialization status. - """ - return self.parsed and self.evaluated and self.finalized + OModel.__init__(self, routine) + # TODO: temporary override for development def parse(self, force=False): self.parsed = True return self.parsed From a6bd8b43ab6e40a16befc7e5c10d895a71df0b50 Mon Sep 17 00:00:00 2001 From: jinningwang Date: Sat, 16 Nov 2024 18:35:27 -0500 Subject: [PATCH 50/96] Add Expression into syms.val_map --- ams/core/symprocessor.py | 1 + 1 file changed, 1 insertion(+) diff --git a/ams/core/symprocessor.py b/ams/core/symprocessor.py index 1cd09167..289083c0 100644 --- a/ams/core/symprocessor.py +++ b/ams/core/symprocessor.py @@ -171,6 +171,7 @@ def generate_symbols(self, force_generate=False): self.sub_map[rf"\b{ename}\b"] = f"self.rtn.{ename}._v" else: self.sub_map[rf"\b{ename}\b"] = f"self.om.{ename}" + self.val_map[rf"\b{ename}\b"] = f"rtn.{ename}.v" self.tex_map[rf"\b{ename}\b"] = f'{expr.tex_name}' # Constraints From 9243ebb9554a192adcf0ba6713e390dee5c73f31 Mon Sep 17 00:00:00 2001 From: jinningwang Date: Sat, 16 Nov 2024 23:23:00 -0500 Subject: [PATCH 51/96] Minor refactor --- ams/nlopt/__init__.py | 5 --- ams/nlopt/pfmodel.py | 53 ------------------------ ams/opt/__init__.py | 3 +- ams/opt/omodel.py | 92 ++++++++++++++++++++++++++++++++++++++++- ams/opt/pfmodel.py | 18 ++++++++ ams/routines/routine.py | 4 +- 6 files changed, 112 insertions(+), 63 deletions(-) delete mode 100644 ams/nlopt/__init__.py delete mode 100644 ams/nlopt/pfmodel.py create mode 100644 ams/opt/pfmodel.py diff --git a/ams/nlopt/__init__.py b/ams/nlopt/__init__.py deleted file mode 100644 index cc850ffe..00000000 --- a/ams/nlopt/__init__.py +++ /dev/null @@ -1,5 +0,0 @@ -""" -Module for non-linear problems. -""" - -from ams.nlopt.pfmodel import PFModel # NOQA diff --git a/ams/nlopt/pfmodel.py b/ams/nlopt/pfmodel.py deleted file mode 100644 index 6ac250f9..00000000 --- a/ams/nlopt/pfmodel.py +++ /dev/null @@ -1,53 +0,0 @@ -""" -Module for non-linear power flow solving. -""" -import logging - -from ams.opt import OModel - - -logger = logging.getLogger(__name__) - - -class PFModel(OModel): - """ - Base class for power flow solver. - """ - - def __init__(self, routine): - OModel.__init__(self, routine) - - # TODO: temporary override for development - def parse(self, force=False): - self.parsed = True - return self.parsed - - def evaluate(self, force=False): - self.evaluated = True - return self.evaluated - - def finalize(self, force=False): - self.finalized = True - return self.finalized - - def init(self, force=False): - """ - Initialize the power flow solver. - """ - self.parse(force) - self.evaluate(force) - self.finalize(force) - return self.initialized - - def update(self, params): - pass - - @property - def class_name(self): - """ - Return the class name - """ - return self.__class__.__name__ - - def __repr__(self) -> str: - return f'{self.rtn.class_name}.{self.__class__.__name__} at {hex(id(self))}' diff --git a/ams/opt/__init__.py b/ams/opt/__init__.py index 9cc99133..accdd76e 100644 --- a/ams/opt/__init__.py +++ b/ams/opt/__init__.py @@ -9,4 +9,5 @@ from ams.opt.constraint import Constraint # NOQA from ams.opt.objective import Objective # NOQA from ams.opt.expression import Expression # NOQA -from ams.opt.omodel import OModel # NOQA +from ams.opt.omodel import OModelBase, OModel # NOQA +from ams.opt.pfmodel import PFModel # NOQA diff --git a/ams/opt/omodel.py b/ams/opt/omodel.py index cda85569..794eda31 100644 --- a/ams/opt/omodel.py +++ b/ams/opt/omodel.py @@ -16,9 +16,9 @@ logger = logging.getLogger(__name__) -class OModel: +class OModelBase: """ - Base class for optimization models. + Template class for optimization models. Parameters ---------- @@ -66,6 +66,94 @@ def initialized(self): """ return self.parsed and self.evaluated and self.finalized + def parse(self, force=False): + self.parsed = True + return self.parsed + + def _evaluate_params(self): + return True + + def _evaluate_vars(self): + return True + + def _evaluate_constrs(self): + return True + + def _evaluate_obj(self): + return True + + def _evaluate_exprs(self): + return True + + def _evaluate_exprcs(self): + return True + + def evaluate(self, force=False): + self._evaluate_params() + self._evaluate_vars() + self._evaluate_exprs() + self._evaluate_constrs() + self._evaluate_obj() + self._evaluate_exprcs() + self.evaluated = True + return self.evaluated + + def finalize(self, force=False): + self.finalized = True + return True + + def init(self, force=False): + self.parse(force) + self.evaluate(force) + self.finalize(force) + return self.initialized + + @property + def class_name(self): + return self.__class__.__name__ + + def _register_attribute(self, key, value): + """ + Register a pair of attributes to OModel instance. + + Called within ``__setattr__``, this is where the magic happens. + Subclass attributes are automatically registered based on the variable type. + """ + if isinstance(value, cp.Variable): + self.vars[key] = value + elif isinstance(value, cp.Constraint): + self.constrs[key] = value + elif isinstance(value, cp.Parameter): + self.params[key] = value + elif isinstance(value, cp.Expression): + self.exprs[key] = value + + def __setattr__(self, name: str, value: Any): + super().__setattr__(name, value) + self._register_attribute(name, value) + + def update(self, params): + return True + + def __repr__(self) -> str: + return f'{self.rtn.class_name}.{self.__class__.__name__} at {hex(id(self))}' + + +class OModel(OModelBase): + """ + Base class for optimization models. + """ + + def __init__(self, routine): + OModelBase.__init__(self, routine) + + @property + def initialized(self): + """ + Return the initialization status. + """ + return self.parsed and self.evaluated and self.finalized + @ensure_symbols def parse(self, force=False): """ diff --git a/ams/opt/pfmodel.py b/ams/opt/pfmodel.py new file mode 100644 index 00000000..e2b08ffa --- /dev/null +++ b/ams/opt/pfmodel.py @@ -0,0 +1,18 @@ +""" +Module for non-linear power flow solving. +""" +import logging + +from ams.opt.omodel import OModelBase + + +logger = logging.getLogger(__name__) + + +class PFModel(OModelBase): + """ + Base class for power flow solver. + """ + + def __init__(self, routine): + OModelBase.__init__(self, routine) diff --git a/ams/routines/routine.py b/ams/routines/routine.py index f4a2cd40..e623f6d0 100644 --- a/ams/routines/routine.py +++ b/ams/routines/routine.py @@ -16,8 +16,8 @@ from ams.core.symprocessor import SymProcessor from ams.core.documenter import RDocumenter from ams.core.service import RBaseService, ValueService -from ams.opt import OModel, Param, Var, Constraint, Objective, ExpressionCalc, Expression -from ams.nlopt import PFModel +from ams.opt import OModel, PFModel +from ams.opt import Param, Var, Constraint, Objective, ExpressionCalc, Expression from ams.shared import pd From 789936398ade5877d242df0d768a16e5b1c09340 Mon Sep 17 00:00:00 2001 From: jinningwang Date: Sat, 16 Nov 2024 23:25:26 -0500 Subject: [PATCH 52/96] Docstring --- ams/opt/omodel.py | 54 ++++++++++++++++++++++++----------------------- 1 file changed, 28 insertions(+), 26 deletions(-) diff --git a/ams/opt/omodel.py b/ams/opt/omodel.py index 794eda31..2781eca3 100644 --- a/ams/opt/omodel.py +++ b/ams/opt/omodel.py @@ -19,32 +19,6 @@ class OModelBase: """ Template class for optimization models. - - Parameters - ---------- - routine: Routine - Routine that to be modeled. - - Attributes - ---------- - prob: cvxpy.Problem - Optimization model. - exprs: OrderedDict - Expressions registry. - params: OrderedDict - Parameters registry. - vars: OrderedDict - Decision variables registry. - constrs: OrderedDict - Constraints registry. - obj: Objective - Objective function. - initialized: bool - Flag indicating if the model is initialized. - parsed: bool - Flag indicating if the model is parsed. - evaluated: bool - Flag indicating if the model is evaluated. """ def __init__(self, routine): @@ -142,6 +116,34 @@ def __repr__(self) -> str: class OModel(OModelBase): """ Base class for optimization models. + + Parameters + ---------- + routine: Routine + Routine that to be modeled. + + Attributes + ---------- + prob: cvxpy.Problem + Optimization model. + exprs: OrderedDict + Expressions registry. + params: OrderedDict + Parameters registry. + vars: OrderedDict + Decision variables registry. + constrs: OrderedDict + Constraints registry. + obj: Objective + Objective function. + initialized: bool + Flag indicating if the model is initialized. + parsed: bool + Flag indicating if the model is parsed. + evaluated: bool + Flag indicating if the model is evaluated. + finalized: bool + Flag indicating if the model is finalized. """ def __init__(self, routine): From 7d47cbe8822d773d660831ddc47c3f8dedc316ec Mon Sep 17 00:00:00 2001 From: jinningwang Date: Sun, 17 Nov 2024 10:45:09 -0500 Subject: [PATCH 53/96] Remove no_parse option for class Expression --- ams/core/symprocessor.py | 5 +---- ams/opt/expression.py | 41 +++++++--------------------------------- 2 files changed, 8 insertions(+), 38 deletions(-) diff --git a/ams/core/symprocessor.py b/ams/core/symprocessor.py index 289083c0..365f7a5e 100644 --- a/ams/core/symprocessor.py +++ b/ams/core/symprocessor.py @@ -167,10 +167,7 @@ def generate_symbols(self, force_generate=False): # Expressions for ename, expr in self.parent.exprs.items(): self.inputs_dict[ename] = sp.symbols(f'{ename}') - if expr.no_parse: - self.sub_map[rf"\b{ename}\b"] = f"self.rtn.{ename}._v" - else: - self.sub_map[rf"\b{ename}\b"] = f"self.om.{ename}" + self.sub_map[rf"\b{ename}\b"] = f"self.om.{ename}" self.val_map[rf"\b{ename}\b"] = f"rtn.{ename}.v" self.tex_map[rf"\b{ename}\b"] = f'{expr.tex_name}' diff --git a/ams/opt/expression.py b/ams/opt/expression.py index 51532bc6..a80696c0 100644 --- a/ams/opt/expression.py +++ b/ams/opt/expression.py @@ -22,10 +22,6 @@ class Expression(OptzBase): """ Base class for expressions used in a routine. - When `no_parse` is True, the Expression will have its own value storage - `_v` and will not be evaluated as a CVXPY expression. - In this case, `model` should be provided to indicate the owner model or group. - Parameters ---------- name : str, optional @@ -40,8 +36,6 @@ class Expression(OptzBase): Unit e_str : str, optional Expression string - no_parse : bool, optional - Do not parse the expression model : str, optional Name of the owner model or group. src : str, optional @@ -58,7 +52,6 @@ def __init__(self, info: Optional[str] = None, unit: Optional[str] = None, e_str: Optional[str] = None, - no_parse: Optional[bool] = False, model: Optional[str] = None, src: Optional[str] = None, vtype: Optional[str] = float, @@ -69,13 +62,10 @@ def __init__(self, self.e_str = e_str self.optz = None self.code = None - self.no_parse = no_parse self.model = model self.owner = None self.src = src self.is_group = False - self._v = None # internal value storage when no_parse is True - self.vtype = vtype # value type, used when no_parse is True self.horizon = horizon def get_idx(self): @@ -119,15 +109,6 @@ def evaluate(self): bool Returns True if the evaluation is successful, False otherwise. """ - # TODO: when no_parse, should initialize _v - if self.no_parse: - if self.horizon is not None: - shape = (self.owner.n, self.horizon.n) - else: - shape = (self.owner.n,) - self._v = np.zeros(shape, dtype=self.vtype) - return True - msg = f" - Expression <{self.name}>: {self.code}" logger.debug(pretty_long_message(msg, _prefix, max_length=_max_length)) try: @@ -158,9 +139,7 @@ def v(self): """ Return the CVXPY expression value. """ - if self.no_parse: - return self._v - elif self.optz is None: + if self.optz is None: return None else: return self.optz.value @@ -170,24 +149,18 @@ def v(self, value): """ Set the value. """ - if self.no_parse: - self._v[...] = value - else: - logger.warning('Cannot set value to an Expression that is not no_parse.') + raise NotImplementedError('Cannot set value to an Expression.') @property def shape(self): """ Return the shape. """ - if self.no_parse: - return self._v.shape - else: - try: - return self.om.__dict__[self.name].shape - except KeyError: - logger.warning('Shape info is not ready before initialization.') - return None + try: + return self.om.__dict__[self.name].shape + except KeyError: + logger.warning('Shape info is not ready before initialization.') + return None @property def size(self): From 93f16685e9fbe391a0fa5568e11031c2ce9f8e44 Mon Sep 17 00:00:00 2001 From: jinningwang Date: Sun, 17 Nov 2024 10:49:13 -0500 Subject: [PATCH 54/96] Update release notes --- docs/source/release-notes.rst | 3 +-- 1 file changed, 1 insertion(+), 2 deletions(-) diff --git a/docs/source/release-notes.rst b/docs/source/release-notes.rst index 38eabdda..8bc5788a 100644 --- a/docs/source/release-notes.rst +++ b/docs/source/release-notes.rst @@ -12,13 +12,12 @@ Pre-v1.0.0 v0.9.12 (202x-xx-xx) -------------------- -TODO: class `Expression`: registry in `OModel` and `Routine`, fit symbol processor, fit documenter - - Refactor `OModel.initialized` as a property method - Add a demo to show using `Constraint.e` for debugging - Fix `ams.opt.omodel.Param.evaluate()` when its value is a number - Improve `ams.opt.omodel.ExpressionCalc()` for efficiency - Refactor module `ams.opt` +- Add class `ams.opt.Expression` RC1 ~~~~ From b2c7f50e8acab5534c3eaa2cb9d5e45483de4520 Mon Sep 17 00:00:00 2001 From: jinningwang Date: Thu, 21 Nov 2024 20:25:17 -0500 Subject: [PATCH 55/96] Remove module ams.opt.pfmodel --- ams/opt/__init__.py | 1 - ams/opt/pfmodel.py | 18 ------------------ ams/routines/routine.py | 6 +++--- 3 files changed, 3 insertions(+), 22 deletions(-) delete mode 100644 ams/opt/pfmodel.py diff --git a/ams/opt/__init__.py b/ams/opt/__init__.py index accdd76e..83b5268f 100644 --- a/ams/opt/__init__.py +++ b/ams/opt/__init__.py @@ -10,4 +10,3 @@ from ams.opt.objective import Objective # NOQA from ams.opt.expression import Expression # NOQA from ams.opt.omodel import OModelBase, OModel # NOQA -from ams.opt.pfmodel import PFModel # NOQA diff --git a/ams/opt/pfmodel.py b/ams/opt/pfmodel.py deleted file mode 100644 index e2b08ffa..00000000 --- a/ams/opt/pfmodel.py +++ /dev/null @@ -1,18 +0,0 @@ -""" -Module for non-linear power flow solving. -""" -import logging - -from ams.opt.omodel import OModelBase - - -logger = logging.getLogger(__name__) - - -class PFModel(OModelBase): - """ - Base class for power flow solver. - """ - - def __init__(self, routine): - OModelBase.__init__(self, routine) diff --git a/ams/routines/routine.py b/ams/routines/routine.py index e623f6d0..9ea1319d 100644 --- a/ams/routines/routine.py +++ b/ams/routines/routine.py @@ -16,7 +16,7 @@ from ams.core.symprocessor import SymProcessor from ams.core.documenter import RDocumenter from ams.core.service import RBaseService, ValueService -from ams.opt import OModel, PFModel +from ams.opt import OModel from ams.opt import Param, Var, Constraint, Objective, ExpressionCalc, Expression from ams.shared import pd @@ -121,8 +121,8 @@ def __init__(self, system=None, config=None): # --- optimization modeling --- # NOTE: for specific routins, the om is a dedicated model - if self.class_name == 'PFlow2': - self.om = PFModel(routine=self) + if self.class_name in ['PFlow2']: + self.om = None else: self.om = OModel(routine=self) # optimization model From 4b0ab69190a60721283eeb876d18fb4ab437d9bf Mon Sep 17 00:00:00 2001 From: jinningwang Date: Thu, 21 Nov 2024 20:25:39 -0500 Subject: [PATCH 56/96] Refactor func to_andes --- ams/interface.py | 34 +++++++++++++++++++++++----------- 1 file changed, 23 insertions(+), 11 deletions(-) diff --git a/ams/interface.py b/ams/interface.py index fe93dd32..a7d1938a 100644 --- a/ams/interface.py +++ b/ams/interface.py @@ -17,6 +17,7 @@ # Models used in ANDES PFlow +# FIXME: add DC models, e.g. Node pflow_dict = OrderedDict([ ('Bus', create_entry('Vn', 'vmax', 'vmin', 'v0', 'a0', 'xcoord', 'ycoord', 'area', 'zone', @@ -49,6 +50,26 @@ 'pq': 'PQ', } +def _to_andes_pflow(system, no_output=False, default_config=True): + """ + Helper function to convert the AMS system to an ANDES system with only + power flow models. + """ + + adsys = andes_System(no_outpu=no_output, default_config=default_config) + # FIXME: is there a systematic way to do this? Other config might be needed + adsys.config.freq = system.config.freq + adsys.config.mva = system.config.mva + + for mdl_name, mdl_cols in pflow_dict.items(): + mdl = getattr(system, mdl_name) + mdl.cache.refresh("df_in") # refresh cache + for row in mdl.cache.df_in[mdl_cols].to_dict(orient='records'): + adsys.add(mdl_name, row) + + return adsys + + def to_andes(system, addfile=None, setup=False, no_output=False, default_config=True, @@ -110,17 +131,8 @@ def to_andes(system, addfile=None, """ t0, _ = elapsed() - adsys = andes_System(no_output=no_output, - default_config=default_config) - # FIXME: is there a systematic way to do this? Other config might be needed - adsys.config.freq = system.config.freq - adsys.config.mva = system.config.mva - - for mdl_name, mdl_cols in pflow_dict.items(): - mdl = getattr(system, mdl_name) - mdl.cache.refresh("df_in") # refresh cache - for row in mdl.cache.df_in[mdl_cols].to_dict(orient='records'): - adsys.add(mdl_name, row) + # --- convert power flow models --- + adsys = _to_andes_pflow(system, no_output=no_output, default_config=default_config) _, s = elapsed(t0) From ea25c96a0f769eaa5f6ae4ffaa29e7558c9ff15e Mon Sep 17 00:00:00 2001 From: jinningwang Date: Thu, 21 Nov 2024 22:36:56 -0500 Subject: [PATCH 57/96] Fix ams.interface._to_andes_pflow kwargs --- ams/interface.py | 9 +++++---- 1 file changed, 5 insertions(+), 4 deletions(-) diff --git a/ams/interface.py b/ams/interface.py index a7d1938a..2f155051 100644 --- a/ams/interface.py +++ b/ams/interface.py @@ -50,13 +50,13 @@ 'pq': 'PQ', } -def _to_andes_pflow(system, no_output=False, default_config=True): +def _to_andes_pflow(system, no_output=False, default_config=True, **kwargs): """ Helper function to convert the AMS system to an ANDES system with only power flow models. """ - adsys = andes_System(no_outpu=no_output, default_config=default_config) + adsys = andes_System(no_outpu=no_output, default_config=default_config, **kwargs) # FIXME: is there a systematic way to do this? Other config might be needed adsys.config.freq = system.config.freq adsys.config.mva = system.config.mva @@ -73,7 +73,8 @@ def _to_andes_pflow(system, no_output=False, default_config=True): def to_andes(system, addfile=None, setup=False, no_output=False, default_config=True, - verify=True, tol=1e-3): + verify=True, tol=1e-3, + **kwargs): """ Convert the AMS system to an ANDES system. @@ -132,7 +133,7 @@ def to_andes(system, addfile=None, t0, _ = elapsed() # --- convert power flow models --- - adsys = _to_andes_pflow(system, no_output=no_output, default_config=default_config) + adsys = _to_andes_pflow(system, no_output=no_output, default_config=default_config, **kwargs) _, s = elapsed(t0) From d5e38fc693c96cf915e499167f70421d54c36466 Mon Sep 17 00:00:00 2001 From: jinningwang Date: Thu, 21 Nov 2024 22:38:06 -0500 Subject: [PATCH 58/96] [WIP] PFlow2, draft using ANDES PFlow routine --- ams/routines/pflow2.py | 283 ++++++++++++++++------------------------- 1 file changed, 107 insertions(+), 176 deletions(-) diff --git a/ams/routines/pflow2.py b/ams/routines/pflow2.py index 475c60f2..4775f4fb 100644 --- a/ams/routines/pflow2.py +++ b/ams/routines/pflow2.py @@ -2,215 +2,146 @@ Power flow routines independent from PYPOWER. """ import logging +from collections import OrderedDict -import numpy as np # noqa -from ams.core.param import RParam -from ams.core.service import NumOp, NumOpDual, VarSelect # noqa +from andes.utils.misc import elapsed from ams.routines.routine import RoutineBase - -from ams.opt import Var, Constraint, Objective, Expression, ExpressionCalc # noqa - +from ams.interface import _to_andes_pflow logger = logging.getLogger(__name__) class PFlow2(RoutineBase): """ - Power flow routine. + Power flow analysis using ANDES PFlow routine. + + More PFlow settings can be changed via `PFlow2._adsys.config` and `PFlow2.om.config`. - [In progress] + Reference + --------- - References - ---------- - 1. R. D. Zimmerman, C. E. Murillo-Sanchez, and R. J. Thomas, “MATPOWER: Steady-State Operations, Planning, and - Analysis Tools for Power Systems Research and Education,” IEEE Trans. Power Syst., vol. 26, no. 1, pp. 12–19, - Feb. 2011 + [1] ANDES Documentation - Simulation and Plot, [Online], + + Available: + + https://docs.andes.app/en/latest/_examples/ex1.html """ def __init__(self, system, config): RoutineBase.__init__(self, system, config) self.info = 'AC Power Flow' self.type = 'PF' + self._adsys = None + + self.config.add(OrderedDict((('tol', 1e-6), + ('max_iter', 25), + ('method', 'NR'), + ('check_conn', 1), + ('n_factorize', 4), + ))) + self.config.add_extra("_help", + tol="convergence tolerance", + max_iter="max. number of iterations", + method="calculation method", + check_conn='check connectivity before power flow', + n_factorize="first N iterations to factorize Jacobian in dishonest method", + ) + self.config.add_extra("_alt", + tol="float", + method=("NR", "dishonest", "NK"), + check_conn=(0, 1), + max_iter=">=10", + n_factorize=">0", + ) + + def init(self, **kwargs): + """ + Initialize the ANDES PFlow routine. - # --- Mapping Section --- - # TODO: skip for now - # # --- from map --- - # self.map1.update({ - # 'ug': ('StaticGen', 'u'), - # }) - # # --- to map --- - # self.map2.update({ - # 'vBus': ('Bus', 'v0'), - # 'ug': ('StaticGen', 'u'), - # 'pg': ('StaticGen', 'p0'), - # }) - - # --- Data Section --- - # --- generator --- - self.ug = RParam(info='Gen connection status', - name='ug', tex_name=r'u_{g}', - model='StaticGen', src='u', - no_parse=True) - # --- load --- - self.pd = RParam(info='active demand', - name='pd', tex_name=r'p_{d}', - model='StaticLoad', src='p0', - unit='p.u.',) - self.qd = RParam(info='reactive demand', - name='qd', tex_name=r'q_{d}', - model='StaticLoad', src='q0', - unit='p.u.',) - # --- line --- - self.ul = RParam(info='Line connection status', - name='ul', tex_name=r'u_{l}', - model='Line', src='u', - no_parse=True,) - # --- shunt --- - self.gsh = RParam(info='shunt conductance', - name='gsh', tex_name=r'g_{sh}', - model='Shunt', src='g', - no_parse=True,) - # --- connection matrix --- - self.Cg = RParam(info='Gen connection matrix', - name='Cg', tex_name=r'C_{g}', - model='mats', src='Cg', - no_parse=True, sparse=True,) - self.Cl = RParam(info='Load connection matrix', - name='Cl', tex_name=r'C_{l}', - model='mats', src='Cl', - no_parse=True, sparse=True,) - self.CftT = RParam(info='Transpose of line connection matrix', - name='CftT', tex_name=r'C_{ft}^T', - model='mats', src='CftT', - no_parse=True, sparse=True,) - self.Csh = RParam(info='Shunt connection matrix', - name='Csh', tex_name=r'C_{sh}', - model='mats', src='Csh', - no_parse=True, sparse=True,) - # --- system matrix --- - self.Bbus = RParam(info='Bus admittance matrix', - name='Bbus', tex_name=r'B_{bus}', - model='mats', src='Bbus', - no_parse=True, sparse=True,) - self.Bf = RParam(info='Bf matrix', - name='Bf', tex_name=r'B_{f}', - model='mats', src='Bf', - no_parse=True, sparse=True,) - self.Pbusinj = RParam(info='Bus power injection vector', - name='Pbusinj', tex_name=r'P_{bus}^{inj}', - model='mats', src='Pbusinj', - no_parse=True,) - self.Pfinj = RParam(info='Line power injection vector', - name='Pfinj', tex_name=r'P_{f}^{inj}', - model='mats', src='Pfinj', - no_parse=True,) - - # --- Model Section --- - # --- generation --- - self.pg = Var(info='Gen active power', - unit='p.u.', - name='pg', tex_name=r'p_g', - model='StaticGen', src='p',) - self.qg = Var(info='Gen reactive power', - unit='p.u.', - name='qg', tex_name=r'q_g', - model='StaticGen', src='q',) - # --- bus --- - self.aBus = Var(info='Bus voltage angle', - unit='rad', - name='aBus', tex_name=r'\theta_{bus}', - model='Bus', src='a',) - self.vBus = Var(info='Bus voltage magnitude', - unit='p.u.', - name='vBus', tex_name=r'v_{bus}', - model='Bus', src='v',) - # --- power balance --- - pb = 'Bbus@aBus + Pbusinj + Cl@pd + Csh@gsh - Cg@pg' - self.pb = Constraint(name='pb', info='power balance', - e_str=pb, is_eq=True, - ) - self.Vc = ExpressionCalc(name='Vc', - info='power balance', - e_str='vBus dot exp(1j * aBus)', - ) - # --- objective --- - # self.obj = Objective(name='obj', - # info='No objective', unit='$', - # sense='min', e_str='(0)',) - - # TODO: we might also need to override self.om.init()? - - # TODO: seems we might need override the following methods - # def init(self, **kwargs): - # """ - # Initialize the routine optimization model. - # args and kwargs go to `self.om.init()` (`cvxpy.Problem.__init__()`). - # """ - # raise NotImplementedError('Not implemented yet.') + kwargs go to andes.system.System(). + """ + self._adsys = _to_andes_pflow(self.system, + no_output=self.system.files.no_output, + config=self.config.as_dict(), + **kwargs) + self._adsys.setup() + self.om = self._adsys.PFlow + self.initialized = True + return self.initialized def solve(self, **kwargs): """ - Solve the routine optimization model. - args and kwargs go to `self.om.prob.solve()` (`cvxpy.Problem.solve()`). + Placeholder. """ - raise NotImplementedError('Not implemented yet.') + return True def run(self, **kwargs): """ Run the routine. - - Following kwargs go to `self.init()`: `force`, `force_mats`, `force_constr`, `force_om`. - - Following kwargs go to `self.solve()`: `solver`, `verbose`, `gp`, `qcp`, `requires_grad`, - `enforce_dpp`, `ignore_dpp`, `method`, and all rest. - - Parameters - ---------- - force : bool, optional - If True, force re-initialization. Defaults to False. - force_mats : bool, optional - If True, force re-generating matrices. Defaults to False. - force_constr : bool, optional - Whether to turn on all constraints. - force_om : bool, optional - If True, force re-generating optimization model. Defaults to False. - solver: str, optional - The solver to use. For example, 'GUROBI', 'ECOS', 'SCS', or 'OSQP'. - verbose : bool, optional - Overrides the default of hiding solver output and prints logging - information describing CVXPY's compilation process. - gp : bool, optional - If True, parses the problem as a disciplined geometric program - instead of a disciplined convex program. - qcp : bool, optional - If True, parses the problem as a disciplined quasiconvex program - instead of a disciplined convex program. - requires_grad : bool, optional - Makes it possible to compute gradients of a solution with respect to Parameters - by calling problem.backward() after solving, or to compute perturbations to the variables - given perturbations to Parameters by calling problem.derivative(). - Gradients are only supported for DCP and DGP problems, not quasiconvex problems. - When computing gradients (i.e., when this argument is True), the problem must satisfy the DPP rules. - enforce_dpp : bool, optional - When True, a DPPError will be thrown when trying to solve a - non-DPP problem (instead of just a warning). - Only relevant for problems involving Parameters. Defaults to False. - ignore_dpp : bool, optional - When True, DPP problems will be treated as non-DPP, which may speed up compilation. Defaults to False. - method : function, optional - A custom solve method to use. """ - raise NotImplementedError('Not implemented yet.') + # --- solve optimization --- + t0, _ = elapsed() + _ = self.om.run() + self.exit_code = self._adsys.exit_code + self.converged = self.exit_code == 0 + _, s = elapsed(t0) + self.exec_time = float(s.split(" ")[0]) + self.unpack() + return True def _post_solve(self): """ - Post-solve calculations. + Placeholder. """ return True def unpack(self, **kwargs): """ - Unpack the results from CVXPY model. + Unpack the results from ANDES PFlow routine. + """ + # TODO: maybe also include the DC devices results + bus_idx = self.system.Bus.idx.v + self.system.Bus.set(src='v', attr='v', idx=bus_idx, + value=self._adsys.Bus.get(src='v', attr='v', idx=bus_idx)) + self.system.Bus.set(src='a', attr='v', idx=bus_idx, + value=self._adsys.Bus.get(src='a', attr='v', idx=bus_idx)) + pv_idx = self.system.PV.idx.v + self.system.PV.set(src='p', attr='v', idx=pv_idx, + value=self.system.PV.get(src='p0', attr='v', idx=pv_idx)) + self.system.PV.set(src='q', attr='v', idx=pv_idx, + value=self._adsys.PV.get(src='q', attr='v', idx=pv_idx)) + slack_idx = self.system.Slack.idx.v + self.system.Slack.set(src='p', attr='v', idx=slack_idx, + value=self._adsys.Slack.get(src='p', attr='v', idx=slack_idx)) + self.system.Slack.set(src='q', attr='v', idx=slack_idx, + value=self._adsys.Slack.get(src='q', attr='v', idx=slack_idx)) + return True + + def update(self, **kwargs): + """ + Placeholder. """ - raise NotImplementedError('Not implemented yet.') + return True + + def enable(self, name): + raise NotImplementedError + + def disable(self, name): + raise NotImplementedError + + def addRParam(self, **kwargs): + raise NotImplementedError + + def addService(self, **kwargs): + raise NotImplementedError + + def addConstrs(self, **kwargs): + raise NotImplementedError + + def addVars(self, **kwargs): + raise NotImplementedError + + def export_csv(self, path=None): + # TODO: customize the CSV output + raise NotImplementedError \ No newline at end of file From 76fdf8f94d2a2f114d1a38544c230d86fcb0c845 Mon Sep 17 00:00:00 2001 From: jinningwang Date: Fri, 22 Nov 2024 06:44:14 -0500 Subject: [PATCH 59/96] Initialize RoutineBase.om as OModel --- ams/routines/routine.py | 6 +----- 1 file changed, 1 insertion(+), 5 deletions(-) diff --git a/ams/routines/routine.py b/ams/routines/routine.py index 9ea1319d..d576007e 100644 --- a/ams/routines/routine.py +++ b/ams/routines/routine.py @@ -120,11 +120,7 @@ def __init__(self, system=None, config=None): self.map2 = OrderedDict() # to ANDES # --- optimization modeling --- - # NOTE: for specific routins, the om is a dedicated model - if self.class_name in ['PFlow2']: - self.om = None - else: - self.om = OModel(routine=self) # optimization model + self.om = OModel(routine=self) # optimization model if config is not None: self.config.load(config) From d81e36e03ca8f6ca4f8053c3fe0a907e8cad6025 Mon Sep 17 00:00:00 2001 From: jinningwang Date: Fri, 22 Nov 2024 07:38:28 -0500 Subject: [PATCH 60/96] In ams.main.config_logger, set ANDES using the same streaming level --- ams/main.py | 2 ++ 1 file changed, 2 insertions(+) diff --git a/ams/main.py b/ams/main.py index 3df5861e..93d987be 100644 --- a/ams/main.py +++ b/ams/main.py @@ -18,6 +18,7 @@ from ._version import get_versions from andes.main import _find_cases +from andes.main import config_logger as ad_config_logger from andes.shared import Pool, Process, coloredlogs, unittest, NCPUS_PHYSICAL from andes.utils.misc import elapsed, is_interactive @@ -71,6 +72,7 @@ def config_logger(stream_level=logging.INFO, *, Original author: Hantao Cui License: GPL3 """ + ad_config_logger(stream_level) lg = logging.getLogger('ams') lg.setLevel(logging.DEBUG) From b47fbc74070225cc56ebf8d6ba30cc1ad1d1447d Mon Sep 17 00:00:00 2001 From: jinningwang Date: Fri, 22 Nov 2024 07:43:17 -0500 Subject: [PATCH 61/96] Minor enhancement on ams.report.write --- ams/report.py | 19 +++++++++++++++---- 1 file changed, 15 insertions(+), 4 deletions(-) diff --git a/ams/report.py b/ams/report.py index ea3e62f6..abc3c0a7 100644 --- a/ams/report.py +++ b/ams/report.py @@ -180,15 +180,26 @@ def write(self): text.append(['']) row_name.append( ['Generation', 'Load']) - if rtn.type == 'ACED': + + if hasattr(rtn, 'pd'): + pd = rtn.pd.v.sum().round(6) + else: + pd = rtn.system.PQ.p0.v.sum().round(6) + if hasattr(rtn, 'qd'): + qd = rtn.qd.v.sum().round(6) + else: + qd = rtn.system.PQ.q0.v.sum().round(6) + + if rtn.type in ['ACED', 'PF']: header.append(['P (p.u.)', 'Q (p.u.)']) - Pcol = [rtn.pg.v.sum().round(6), rtn.pd.v.sum().round(6)] - Qcol = [rtn.qg.v.sum().round(6), rtn.qd.v.sum().round(6)] + Pcol = [rtn.pg.v.sum().round(6), pd] + Qcol = [rtn.qg.v.sum().round(6), qd] data.append([Pcol, Qcol]) else: header.append(['P (p.u.)']) - Pcol = [rtn.pg.v.sum().round(6), rtn.pd.v.sum().round(6)] + Pcol = [rtn.pg.v.sum().round(6), pd] data.append([Pcol]) + # --- routine data --- text.extend(text_sum) header.extend(header_sum) From 5bf77f9b1cf6157b53232a47cbb917c729895882 Mon Sep 17 00:00:00 2001 From: jinningwang Date: Fri, 22 Nov 2024 07:44:29 -0500 Subject: [PATCH 62/96] PFlow2, interface ANDES PFlow routine --- ams/routines/pflow2.py | 71 ++++++++++++++++++++++++++++-------------- 1 file changed, 48 insertions(+), 23 deletions(-) diff --git a/ams/routines/pflow2.py b/ams/routines/pflow2.py index 4775f4fb..576c0701 100644 --- a/ams/routines/pflow2.py +++ b/ams/routines/pflow2.py @@ -7,6 +7,7 @@ from andes.utils.misc import elapsed from ams.routines.routine import RoutineBase +from ams.opt import Var from ams.interface import _to_andes_pflow logger = logging.getLogger(__name__) @@ -16,12 +17,16 @@ class PFlow2(RoutineBase): """ Power flow analysis using ANDES PFlow routine. - More PFlow settings can be changed via `PFlow2._adsys.config` and `PFlow2.om.config`. + More settings can be changed via `PFlow2._adsys.config` and `PFlow2._adsys.PFlow.config`. + + All generator output powers, bus voltages, and angles are included in the variable definitions. + However, not all of these are unknowns; the definitions are provided for easy access. Reference --------- - [1] ANDES Documentation - Simulation and Plot, [Online], + [1] M. L. Crow, Computational methods for electric power systems. 2015. + [2] ANDES Documentation - Simulation and Plot, [Online], Available: @@ -55,6 +60,23 @@ def __init__(self, system, config): n_factorize=">0", ) + self.pg = Var(info='Gen active power', + unit='p.u.', + name='pg', tex_name=r'p_g', + model='StaticGen', src='p') + self.qg = Var(info='Gen reactive power', + unit='p.u.', + name='qg', tex_name=r'q_g', + model='StaticGen', src='q') + self.aBus = Var(info='Bus voltage angle', + unit='rad', + name='aBus', tex_name=r'\theta_{bus}', + model='Bus', src='a',) + self.vBus = Var(info='Bus voltage magnitude', + unit='p.u.', + name='vBus', tex_name=r'V_{bus}', + model='Bus', src='v',) + def init(self, **kwargs): """ Initialize the ANDES PFlow routine. @@ -66,7 +88,7 @@ def init(self, **kwargs): config=self.config.as_dict(), **kwargs) self._adsys.setup() - self.om = self._adsys.PFlow + self.om.init() self.initialized = True return self.initialized @@ -82,7 +104,7 @@ def run(self, **kwargs): """ # --- solve optimization --- t0, _ = elapsed() - _ = self.om.run() + _ = self._adsys.PFlow.run() self.exit_code = self._adsys.exit_code self.converged = self.exit_code == 0 _, s = elapsed(t0) @@ -101,21 +123,28 @@ def unpack(self, **kwargs): Unpack the results from ANDES PFlow routine. """ # TODO: maybe also include the DC devices results - bus_idx = self.system.Bus.idx.v - self.system.Bus.set(src='v', attr='v', idx=bus_idx, - value=self._adsys.Bus.get(src='v', attr='v', idx=bus_idx)) - self.system.Bus.set(src='a', attr='v', idx=bus_idx, - value=self._adsys.Bus.get(src='a', attr='v', idx=bus_idx)) - pv_idx = self.system.PV.idx.v - self.system.PV.set(src='p', attr='v', idx=pv_idx, - value=self.system.PV.get(src='p0', attr='v', idx=pv_idx)) - self.system.PV.set(src='q', attr='v', idx=pv_idx, - value=self._adsys.PV.get(src='q', attr='v', idx=pv_idx)) - slack_idx = self.system.Slack.idx.v - self.system.Slack.set(src='p', attr='v', idx=slack_idx, - value=self._adsys.Slack.get(src='p', attr='v', idx=slack_idx)) - self.system.Slack.set(src='q', attr='v', idx=slack_idx, - value=self._adsys.Slack.get(src='q', attr='v', idx=slack_idx)) + sys = self.system + # --- device results --- + bus_idx = sys.Bus.idx.v + sys.Bus.set(src='v', attr='v', idx=bus_idx, + value=self._adsys.Bus.get(src='v', attr='v', idx=bus_idx)) + sys.Bus.set(src='a', attr='v', idx=bus_idx, + value=self._adsys.Bus.get(src='a', attr='v', idx=bus_idx)) + pv_idx = sys.PV.idx.v + sys.PV.set(src='p', attr='v', idx=pv_idx, + value=sys.PV.get(src='p0', attr='v', idx=pv_idx)) + sys.PV.set(src='q', attr='v', idx=pv_idx, + value=self._adsys.PV.get(src='q', attr='v', idx=pv_idx)) + slack_idx = sys.Slack.idx.v + sys.Slack.set(src='p', attr='v', idx=slack_idx, + value=self._adsys.Slack.get(src='p', attr='v', idx=slack_idx)) + sys.Slack.set(src='q', attr='v', idx=slack_idx, + value=self._adsys.Slack.get(src='q', attr='v', idx=slack_idx)) + # --- routine results --- + self.pg.optz.value = sys.StaticGen.get(src='p', attr='v', idx=self.pg.get_idx()) + self.qg.optz.value = sys.StaticGen.get(src='q', attr='v', idx=self.qg.get_idx()) + self.aBus.optz.value = sys.Bus.get(src='a', attr='v', idx=self.aBus.get_idx()) + self.vBus.optz.value = sys.Bus.get(src='v', attr='v', idx=self.vBus.get_idx()) return True def update(self, **kwargs): @@ -141,7 +170,3 @@ def addConstrs(self, **kwargs): def addVars(self, **kwargs): raise NotImplementedError - - def export_csv(self, path=None): - # TODO: customize the CSV output - raise NotImplementedError \ No newline at end of file From 5eaae115c1d4224664058364f482638942e80c0d Mon Sep 17 00:00:00 2001 From: jinningwang Date: Fri, 22 Nov 2024 08:02:30 -0500 Subject: [PATCH 63/96] [WIP] DCPF --- ams/routines/dcpf2.py | 51 +++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 51 insertions(+) create mode 100644 ams/routines/dcpf2.py diff --git a/ams/routines/dcpf2.py b/ams/routines/dcpf2.py new file mode 100644 index 00000000..fd67986c --- /dev/null +++ b/ams/routines/dcpf2.py @@ -0,0 +1,51 @@ +""" +Power flow routines. +""" +import logging + +from andes.shared import deg2rad +from andes.utils.misc import elapsed + +from ams.opt import Var +from ams.pypower import runpf +from ams.pypower.core import ppoption +from ams.routines.dcopf import DCOPF + +from ams.io.pypower import system2ppc +from ams.core.param import RParam + +logger = logging.getLogger(__name__) + + +class DCPF2(DCOPF): + """ + DC power flow, overload the ``solve``, ``unpack``, and ``run`` methods. + + Notes + ----- + 1. DCPF is solved with PYPOWER ``runpf`` function. + 2. DCPF formulation is not complete yet, but this does not affect the + results because the data are passed to PYPOWER for solving. + """ + + def __init__(self, system, config): + DCOPF.__init__(self, system, config) + self.info = 'DC Power Flow' + self.type = 'PF' + + self.obj.e_str = '0' + + def summary(self, **kwargs): + """ + # TODO: Print power flow summary. + """ + raise NotImplementedError + + def enable(self, name): + raise NotImplementedError + + def disable(self, name): + raise NotImplementedError + + def dc2ac(self, name): + raise NotImplementedError From 9eb8f31a4e0e260d954d5d1aee913c807250e550 Mon Sep 17 00:00:00 2001 From: jinningwang Date: Fri, 22 Nov 2024 15:25:43 -0500 Subject: [PATCH 64/96] Switch from PYPOWER PFlow to ANDES PFlow --- ams/interface.py | 23 ++++ ams/routines/__init__.py | 2 +- ams/routines/pflow.py | 252 ++++++++++++++++++++++++++------------- ams/routines/pflow0.py | 111 +++++++++++++++++ ams/routines/pflow2.py | 172 -------------------------- tests/test_interop.py | 4 +- 6 files changed, 305 insertions(+), 259 deletions(-) create mode 100644 ams/routines/pflow0.py delete mode 100644 ams/routines/pflow2.py diff --git a/ams/interface.py b/ams/interface.py index 2f155051..1da15d5b 100644 --- a/ams/interface.py +++ b/ams/interface.py @@ -50,6 +50,27 @@ 'pq': 'PQ', } +def _sync_adsys(amsys, adsys): + """ + Helper function to sync parameters value from AMS to ANDES. + """ + for mname, params in pflow_dict.items(): + ad_mdl = adsys.__dict__[mname] + am_mdl = amsys.__dict__[mname] + idx = am_mdl.idx.v + for param in params: + if param in ['idx', 'name']: + continue + # NOTE: when setting list values to DataParam, sometimes run into error + try: + ad_mdl.set(src=param, attr='v', idx=idx, + value=am_mdl.get(src=param, attr='v', idx=idx)) + except Exception: + logger.debug(f"Skip updating {mname}.{param}") + continue + return adsys + + def _to_andes_pflow(system, no_output=False, default_config=True, **kwargs): """ Helper function to convert the AMS system to an ANDES system with only @@ -67,6 +88,8 @@ def _to_andes_pflow(system, no_output=False, default_config=True, **kwargs): for row in mdl.cache.df_in[mdl_cols].to_dict(orient='records'): adsys.add(mdl_name, row) + adsys = _sync_adsys(amsys=system, adsys=adsys) + return adsys diff --git a/ams/routines/__init__.py b/ams/routines/__init__.py index 87615654..32df57cc 100644 --- a/ams/routines/__init__.py +++ b/ams/routines/__init__.py @@ -8,7 +8,6 @@ all_routines = OrderedDict([ ('dcpf', ['DCPF']), ('pflow', ['PFlow']), - ('pflow2', ['PFlow2']), ('cpf', ['CPF']), ('acopf', ['ACOPF']), ('dcopf', ['DCOPF']), @@ -16,6 +15,7 @@ ('rted', ['RTED', 'RTEDDG', 'RTEDES', 'RTEDVIS']), ('uc', ['UC', 'UCDG', 'UCES']), ('dopf', ['DOPF', 'DOPFVIS']), + ('pflow0', ['PFlow0']), ]) class_names = list_flatten(list(all_routines.values())) diff --git a/ams/routines/pflow.py b/ams/routines/pflow.py index 1f912765..3fba0ed6 100644 --- a/ams/routines/pflow.py +++ b/ams/routines/pflow.py @@ -1,111 +1,197 @@ """ -Power flow routines. +Power flow routines independent from PYPOWER. """ import logging from collections import OrderedDict -from ams.pypower import runpf +from andes.utils.misc import elapsed -from ams.io.pypower import system2ppc -from ams.pypower.core import ppoption from ams.core.param import RParam - -from ams.routines.dcpf import DCPF -from ams.opt import Var +from ams.routines.routine import RoutineBase +from ams.opt import Var, Expression +from ams.interface import _to_andes_pflow, _sync_adsys logger = logging.getLogger(__name__) -class PFlow(DCPF): +class PFlow(RoutineBase): """ - AC Power Flow routine. - - Notes - ----- - 1. AC pwoer flow is solved with PYPOWER ``runpf`` function. - 2. AC power flow formulation in AMS style is NOT DONE YET, - but this does not affect the results - because the data are passed to PYPOWER for solving. + Power flow analysis using ANDES PFlow routine. + + More settings can be changed via `PFlow2._adsys.config` and `PFlow2._adsys.PFlow.config`. + + All generator output powers, bus voltages, and angles are included in the variable definitions. + However, not all of these are unknowns; the definitions are provided for easy access. + + Reference + --------- + + [1] M. L. Crow, Computational methods for electric power systems. 2015. + [2] ANDES Documentation - Simulation and Plot. [Online]. + + Available: + + https://docs.andes.app/en/latest/_examples/ex1.html """ def __init__(self, system, config): - DCPF.__init__(self, system, config) - self.info = "AC Power Flow" - self.type = "PF" - - self.config.add(OrderedDict((('qlim', 0), + RoutineBase.__init__(self, system, config) + self.info = 'AC Power Flow' + self.type = 'PF' + self._adsys = None + + self.config.add(OrderedDict((('tol', 1e-6), + ('max_iter', 25), + ('method', 'NR'), + ('check_conn', 1), + ('n_factorize', 4), ))) self.config.add_extra("_help", - qlim="Enforce generator q limits", + tol="convergence tolerance", + max_iter="max. number of iterations", + method="calculation method", + check_conn='check connectivity before power flow', + n_factorize="first N iterations to factorize Jacobian in dishonest method", ) self.config.add_extra("_alt", - qlim=(0, 1, 2), + tol="float", + method=("NR", "dishonest", "NK"), + check_conn=(0, 1), + max_iter=">=10", + n_factorize=">0", ) - self.qd = RParam(info="reactive power load in system base", - name="qd", tex_name=r"q_{d}", - unit="p.u.", - model="StaticLoad", src="q0",) - - # --- bus --- - self.vBus = Var(info="bus voltage magnitude", - unit="p.u.", - name="vBus", tex_name=r"v_{Bus}", - model="Bus", src="v",) - # --- gen --- - self.qg = Var(info="reactive power generation", - unit="p.u.", - name="qg", tex_name=r"q_{g}", - model="StaticGen", src="q",) - # NOTE: omit AC power flow formulation here - - def solve(self, method="newton", **kwargs): + self.Bf = RParam(info='Bf matrix', + name='Bf', tex_name=r'B_{f}', + model='mats', src='Bf', + no_parse=True, sparse=True,) + self.Pfinj = RParam(info='Line power injection vector', + name='Pfinj', tex_name=r'P_{f}^{inj}', + model='mats', src='Pfinj', + no_parse=True,) + + self.pg = Var(info='Gen active power', + unit='p.u.', + name='pg', tex_name=r'p_g', + model='StaticGen', src='p') + self.qg = Var(info='Gen reactive power', + unit='p.u.', + name='qg', tex_name=r'q_g', + model='StaticGen', src='q') + self.aBus = Var(info='Bus voltage angle', + unit='rad', + name='aBus', tex_name=r'\theta_{bus}', + model='Bus', src='a',) + self.vBus = Var(info='Bus voltage magnitude', + unit='p.u.', + name='vBus', tex_name=r'V_{bus}', + model='Bus', src='v',) + self.plf = Expression(info='Line flow', + name='plf', tex_name=r'p_{lf}', + unit='p.u.', + e_str='Bf@aBus + Pfinj', + model='Line', src=None,) + + def init(self, **kwargs): """ - Solve the AC power flow using PYPOWER. + Initialize the ANDES PFlow routine. + + kwargs go to andes.system.System(). + """ + self._adsys = _to_andes_pflow(self.system, + no_output=self.system.files.no_output, + config=self.config.as_dict(), + **kwargs) + self._adsys.setup() + self.om.init() + self.initialized = True + return self.initialized + + def solve(self, **kwargs): """ - ppc = system2ppc(self.system) + Placeholder. + """ + return True - method_map = dict(newton=1, fdxb=2, fdbx=3, gauss=4) - alg = method_map.get(method) - if alg == 4: - msg = "Gauss method is not fully tested yet, not recommended!" - logger.warning(msg) - if alg is None: - msg = f"Invalid method `{method}` for PFlow." - raise ValueError(msg) - ppopt = ppoption(PF_ALG=alg, ENFORCE_Q_LIMS=self.config.qlim, **kwargs) + def run(self, **kwargs): + """ + Run the routine. + """ + if not self.initialized: + self.init() - res, sstats = runpf(casedata=ppc, ppopt=ppopt) - return res, sstats + t0, _ = elapsed() + _ = self._adsys.PFlow.run() + self.exit_code = self._adsys.exit_code + self.converged = self.exit_code == 0 + _, s = elapsed(t0) + self.exec_time = float(s.split(" ")[0]) - def run(self, **kwargs): + self.unpack() + return True + + def _post_solve(self): + """ + Placeholder. + """ + return True + + def unpack(self, **kwargs): """ - Run AC power flow using PYPOWER. - - Currently, four methods are supported: 'newton', 'fdxb', 'fdbx', 'gauss', - for Newton's method, fast-decoupled, XB, fast-decoupled, BX, and Gauss-Seidel, - respectively. - - Note that gauss method is not recommended because it seems to be much - more slower than the other three methods and not fully tested yet. - - Examples - -------- - >>> ss = ams.load(ams.get_case('matpower/case14.m')) - >>> ss.PFlow.run() - - Parameters - ---------- - force_init : bool - Force initialization. - no_code : bool - Disable showing code. - method : str - Method for solving the power flow. - - Returns - ------- - exit_code : int - Exit code of the routine. + Unpack the results from ANDES PFlow routine. """ - return super().run(**kwargs,) + # TODO: maybe also include the DC devices results + sys = self.system + # --- device results --- + bus_idx = sys.Bus.idx.v + sys.Bus.set(src='v', attr='v', idx=bus_idx, + value=self._adsys.Bus.get(src='v', attr='v', idx=bus_idx)) + sys.Bus.set(src='a', attr='v', idx=bus_idx, + value=self._adsys.Bus.get(src='a', attr='v', idx=bus_idx)) + pv_idx = sys.PV.idx.v + pv_u = sys.PV.get(src='u', attr='v', idx=pv_idx) + # NOTE: for p, we should consider the online status as p0 is a param + sys.PV.set(src='p', attr='v', idx=pv_idx, + value=pv_u * sys.PV.get(src='p0', attr='v', idx=pv_idx)) + sys.PV.set(src='q', attr='v', idx=pv_idx, + value=self._adsys.PV.get(src='q', attr='v', idx=pv_idx)) + slack_idx = sys.Slack.idx.v + sys.Slack.set(src='p', attr='v', idx=slack_idx, + value=self._adsys.Slack.get(src='p', attr='v', idx=slack_idx)) + sys.Slack.set(src='q', attr='v', idx=slack_idx, + value=self._adsys.Slack.get(src='q', attr='v', idx=slack_idx)) + # --- routine results --- + self.pg.optz.value = sys.StaticGen.get(src='p', attr='v', idx=self.pg.get_idx()) + self.qg.optz.value = sys.StaticGen.get(src='q', attr='v', idx=self.qg.get_idx()) + self.aBus.optz.value = sys.Bus.get(src='a', attr='v', idx=self.aBus.get_idx()) + self.vBus.optz.value = sys.Bus.get(src='v', attr='v', idx=self.vBus.get_idx()) + return True + + def update(self, **kwargs): + """ + Placeholder. + """ + if not self.initialized: + self.init() + + _sync_adsys(self.system, self._adsys) + + return True + + def enable(self, name): + raise NotImplementedError + + def disable(self, name): + raise NotImplementedError + + def addRParam(self, **kwargs): + raise NotImplementedError + + def addService(self, **kwargs): + raise NotImplementedError + + def addConstrs(self, **kwargs): + raise NotImplementedError + + def addVars(self, **kwargs): + raise NotImplementedError diff --git a/ams/routines/pflow0.py b/ams/routines/pflow0.py new file mode 100644 index 00000000..8dd72c15 --- /dev/null +++ b/ams/routines/pflow0.py @@ -0,0 +1,111 @@ +""" +Power flow routines. +""" +import logging +from collections import OrderedDict + +from ams.pypower import runpf + +from ams.io.pypower import system2ppc +from ams.pypower.core import ppoption +from ams.core.param import RParam + +from ams.routines.dcpf import DCPF +from ams.opt import Var + +logger = logging.getLogger(__name__) + + +class PFlow0(DCPF): + """ + AC Power Flow routine. + + Notes + ----- + 1. AC pwoer flow is solved with PYPOWER ``runpf`` function. + 2. AC power flow formulation in AMS style is NOT DONE YET, + but this does not affect the results + because the data are passed to PYPOWER for solving. + """ + + def __init__(self, system, config): + DCPF.__init__(self, system, config) + self.info = "AC Power Flow" + self.type = "PF" + + self.config.add(OrderedDict((('qlim', 0), + ))) + self.config.add_extra("_help", + qlim="Enforce generator q limits", + ) + self.config.add_extra("_alt", + qlim=(0, 1, 2), + ) + + self.qd = RParam(info="reactive power load in system base", + name="qd", tex_name=r"q_{d}", + unit="p.u.", + model="StaticLoad", src="q0",) + + # --- bus --- + self.vBus = Var(info="bus voltage magnitude", + unit="p.u.", + name="vBus", tex_name=r"v_{Bus}", + model="Bus", src="v",) + # --- gen --- + self.qg = Var(info="reactive power generation", + unit="p.u.", + name="qg", tex_name=r"q_{g}", + model="StaticGen", src="q",) + # NOTE: omit AC power flow formulation here + + def solve(self, method="newton", **kwargs): + """ + Solve the AC power flow using PYPOWER. + """ + ppc = system2ppc(self.system) + + method_map = dict(newton=1, fdxb=2, fdbx=3, gauss=4) + alg = method_map.get(method) + if alg == 4: + msg = "Gauss method is not fully tested yet, not recommended!" + logger.warning(msg) + if alg is None: + msg = f"Invalid method `{method}` for PFlow." + raise ValueError(msg) + ppopt = ppoption(PF_ALG=alg, ENFORCE_Q_LIMS=self.config.qlim, **kwargs) + + res, sstats = runpf(casedata=ppc, ppopt=ppopt) + return res, sstats + + def run(self, **kwargs): + """ + Run AC power flow using PYPOWER. + + Currently, four methods are supported: 'newton', 'fdxb', 'fdbx', 'gauss', + for Newton's method, fast-decoupled, XB, fast-decoupled, BX, and Gauss-Seidel, + respectively. + + Note that gauss method is not recommended because it seems to be much + more slower than the other three methods and not fully tested yet. + + Examples + -------- + >>> ss = ams.load(ams.get_case('matpower/case14.m')) + >>> ss.PFlow.run() + + Parameters + ---------- + force_init : bool + Force initialization. + no_code : bool + Disable showing code. + method : str + Method for solving the power flow. + + Returns + ------- + exit_code : int + Exit code of the routine. + """ + return super().run(**kwargs,) diff --git a/ams/routines/pflow2.py b/ams/routines/pflow2.py deleted file mode 100644 index 576c0701..00000000 --- a/ams/routines/pflow2.py +++ /dev/null @@ -1,172 +0,0 @@ -""" -Power flow routines independent from PYPOWER. -""" -import logging -from collections import OrderedDict - -from andes.utils.misc import elapsed - -from ams.routines.routine import RoutineBase -from ams.opt import Var -from ams.interface import _to_andes_pflow - -logger = logging.getLogger(__name__) - - -class PFlow2(RoutineBase): - """ - Power flow analysis using ANDES PFlow routine. - - More settings can be changed via `PFlow2._adsys.config` and `PFlow2._adsys.PFlow.config`. - - All generator output powers, bus voltages, and angles are included in the variable definitions. - However, not all of these are unknowns; the definitions are provided for easy access. - - Reference - --------- - - [1] M. L. Crow, Computational methods for electric power systems. 2015. - [2] ANDES Documentation - Simulation and Plot, [Online], - - Available: - - https://docs.andes.app/en/latest/_examples/ex1.html - """ - - def __init__(self, system, config): - RoutineBase.__init__(self, system, config) - self.info = 'AC Power Flow' - self.type = 'PF' - self._adsys = None - - self.config.add(OrderedDict((('tol', 1e-6), - ('max_iter', 25), - ('method', 'NR'), - ('check_conn', 1), - ('n_factorize', 4), - ))) - self.config.add_extra("_help", - tol="convergence tolerance", - max_iter="max. number of iterations", - method="calculation method", - check_conn='check connectivity before power flow', - n_factorize="first N iterations to factorize Jacobian in dishonest method", - ) - self.config.add_extra("_alt", - tol="float", - method=("NR", "dishonest", "NK"), - check_conn=(0, 1), - max_iter=">=10", - n_factorize=">0", - ) - - self.pg = Var(info='Gen active power', - unit='p.u.', - name='pg', tex_name=r'p_g', - model='StaticGen', src='p') - self.qg = Var(info='Gen reactive power', - unit='p.u.', - name='qg', tex_name=r'q_g', - model='StaticGen', src='q') - self.aBus = Var(info='Bus voltage angle', - unit='rad', - name='aBus', tex_name=r'\theta_{bus}', - model='Bus', src='a',) - self.vBus = Var(info='Bus voltage magnitude', - unit='p.u.', - name='vBus', tex_name=r'V_{bus}', - model='Bus', src='v',) - - def init(self, **kwargs): - """ - Initialize the ANDES PFlow routine. - - kwargs go to andes.system.System(). - """ - self._adsys = _to_andes_pflow(self.system, - no_output=self.system.files.no_output, - config=self.config.as_dict(), - **kwargs) - self._adsys.setup() - self.om.init() - self.initialized = True - return self.initialized - - def solve(self, **kwargs): - """ - Placeholder. - """ - return True - - def run(self, **kwargs): - """ - Run the routine. - """ - # --- solve optimization --- - t0, _ = elapsed() - _ = self._adsys.PFlow.run() - self.exit_code = self._adsys.exit_code - self.converged = self.exit_code == 0 - _, s = elapsed(t0) - self.exec_time = float(s.split(" ")[0]) - self.unpack() - return True - - def _post_solve(self): - """ - Placeholder. - """ - return True - - def unpack(self, **kwargs): - """ - Unpack the results from ANDES PFlow routine. - """ - # TODO: maybe also include the DC devices results - sys = self.system - # --- device results --- - bus_idx = sys.Bus.idx.v - sys.Bus.set(src='v', attr='v', idx=bus_idx, - value=self._adsys.Bus.get(src='v', attr='v', idx=bus_idx)) - sys.Bus.set(src='a', attr='v', idx=bus_idx, - value=self._adsys.Bus.get(src='a', attr='v', idx=bus_idx)) - pv_idx = sys.PV.idx.v - sys.PV.set(src='p', attr='v', idx=pv_idx, - value=sys.PV.get(src='p0', attr='v', idx=pv_idx)) - sys.PV.set(src='q', attr='v', idx=pv_idx, - value=self._adsys.PV.get(src='q', attr='v', idx=pv_idx)) - slack_idx = sys.Slack.idx.v - sys.Slack.set(src='p', attr='v', idx=slack_idx, - value=self._adsys.Slack.get(src='p', attr='v', idx=slack_idx)) - sys.Slack.set(src='q', attr='v', idx=slack_idx, - value=self._adsys.Slack.get(src='q', attr='v', idx=slack_idx)) - # --- routine results --- - self.pg.optz.value = sys.StaticGen.get(src='p', attr='v', idx=self.pg.get_idx()) - self.qg.optz.value = sys.StaticGen.get(src='q', attr='v', idx=self.qg.get_idx()) - self.aBus.optz.value = sys.Bus.get(src='a', attr='v', idx=self.aBus.get_idx()) - self.vBus.optz.value = sys.Bus.get(src='v', attr='v', idx=self.vBus.get_idx()) - return True - - def update(self, **kwargs): - """ - Placeholder. - """ - return True - - def enable(self, name): - raise NotImplementedError - - def disable(self, name): - raise NotImplementedError - - def addRParam(self, **kwargs): - raise NotImplementedError - - def addService(self, **kwargs): - raise NotImplementedError - - def addConstrs(self, **kwargs): - raise NotImplementedError - - def addVars(self, **kwargs): - raise NotImplementedError diff --git a/tests/test_interop.py b/tests/test_interop.py index 11070f21..5dd8e709 100644 --- a/tests/test_interop.py +++ b/tests/test_interop.py @@ -127,9 +127,7 @@ def test_verify_pf(self): sa = to_andes(sp, setup=True, no_output=True, default_config=True, verify=False, tol=1e-3) - # NOTE: it is known that there is 1e-7~1e-6 diff in case300.m - self.assertFalse(verify_pf(amsys=sp, adsys=sa, tol=1e-6)) - self.assertTrue(verify_pf(amsys=sp, adsys=sa, tol=1e-3)) + self.assertTrue(verify_pf(amsys=sp, adsys=sa, tol=1e-7)) class TestDataExchange(unittest.TestCase): From 924b7ff8fc743bd5e5db5a938b7e8a05d7758d57 Mon Sep 17 00:00:00 2001 From: jinningwang Date: Fri, 22 Nov 2024 15:28:58 -0500 Subject: [PATCH 65/96] Rename func _sync_adsys as sync_adsys --- ams/interface.py | 20 ++++++++++++++++---- ams/routines/pflow.py | 4 ++-- 2 files changed, 18 insertions(+), 6 deletions(-) diff --git a/ams/interface.py b/ams/interface.py index 1da15d5b..d019d752 100644 --- a/ams/interface.py +++ b/ams/interface.py @@ -50,9 +50,21 @@ 'pq': 'PQ', } -def _sync_adsys(amsys, adsys): +def sync_adsys(amsys, adsys): """ - Helper function to sync parameters value from AMS to ANDES. + Sync parameters value of PFlow models between AMS and ANDES systems. + + Parameters + ---------- + amsys : AMS.system.System + The AMS system. + adsys : ANDES.system.System + The ANDES system. + + Returns + ------- + bool + True if successful. """ for mname, params in pflow_dict.items(): ad_mdl = adsys.__dict__[mname] @@ -68,7 +80,7 @@ def _sync_adsys(amsys, adsys): except Exception: logger.debug(f"Skip updating {mname}.{param}") continue - return adsys + return True def _to_andes_pflow(system, no_output=False, default_config=True, **kwargs): @@ -88,7 +100,7 @@ def _to_andes_pflow(system, no_output=False, default_config=True, **kwargs): for row in mdl.cache.df_in[mdl_cols].to_dict(orient='records'): adsys.add(mdl_name, row) - adsys = _sync_adsys(amsys=system, adsys=adsys) + sync_adsys(amsys=system, adsys=adsys) return adsys diff --git a/ams/routines/pflow.py b/ams/routines/pflow.py index 3fba0ed6..b029d5c0 100644 --- a/ams/routines/pflow.py +++ b/ams/routines/pflow.py @@ -9,7 +9,7 @@ from ams.core.param import RParam from ams.routines.routine import RoutineBase from ams.opt import Var, Expression -from ams.interface import _to_andes_pflow, _sync_adsys +from ams.interface import _to_andes_pflow, sync_adsys logger = logging.getLogger(__name__) @@ -174,7 +174,7 @@ def update(self, **kwargs): if not self.initialized: self.init() - _sync_adsys(self.system, self._adsys) + sync_adsys(self.system, self._adsys) return True From 2df29275f3cef28b9a0eddb881d9ff4d0b21a92c Mon Sep 17 00:00:00 2001 From: jinningwang Date: Fri, 22 Nov 2024 15:32:12 -0500 Subject: [PATCH 66/96] Typo --- ams/__init__.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/ams/__init__.py b/ams/__init__.py index b05d4e9e..379d7b01 100644 --- a/ams/__init__.py +++ b/ams/__init__.py @@ -11,4 +11,4 @@ __author__ = 'Jining Wang' -__all__ = ['System', 'get_case', 'System'] +__all__ = ['System', 'get_case'] From 635cc1a73517a84ba1529db98c9c4fa707cae580 Mon Sep 17 00:00:00 2001 From: jinningwang Date: Fri, 22 Nov 2024 15:45:21 -0500 Subject: [PATCH 67/96] Skip sutogen_stale when instantiating an example ANDES system --- ams/shared.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/ams/shared.py b/ams/shared.py index 9d74670e..92aa5978 100644 --- a/ams/shared.py +++ b/ams/shared.py @@ -23,7 +23,7 @@ pd = LazyImport('import pandas as pd') # --- an empty ANDES system --- -empty_adsys = adSystem() +empty_adsys = adSystem(autogen_stale=False) ad_models = list(empty_adsys.models.keys()) # --- NumPy constants --- From 6d6f3d262724aae3ea1e9dc311cd581b5a9341a4 Mon Sep 17 00:00:00 2001 From: jinningwang Date: Fri, 22 Nov 2024 16:14:08 -0500 Subject: [PATCH 68/96] Adapt module omodel for PFlow --- ams/opt/omodel.py | 23 +++++++++++------------ ams/routines/__init__.py | 1 + 2 files changed, 12 insertions(+), 12 deletions(-) diff --git a/ams/opt/omodel.py b/ams/opt/omodel.py index 2781eca3..79f5f343 100644 --- a/ams/opt/omodel.py +++ b/ams/opt/omodel.py @@ -204,16 +204,15 @@ def parse(self, force=False): val.parse() # --- parse objective functions --- - if self.rtn.type != 'PF': - if self.rtn.obj is not None: - try: - self.rtn.obj.parse() - except Exception as e: - raise Exception(f"Failed to parse Objective <{self.rtn.obj.name}>.\n{e}") - else: - logger.warning(f"{self.rtn.class_name} has no objective function!") - self.parsed = False - return self.parsed + if self.rtn.obj is not None: + try: + self.rtn.obj.parse() + except Exception as e: + raise Exception(f"Failed to parse Objective <{self.rtn.obj.name}>.\n{e}") + elif not self.rtn.class_name in ['PFlow']: + logger.warning(f"{self.rtn.class_name} has no objective function!") + self.parsed = False + return self.parsed # --- parse expressions --- for key, val in self.rtn.exprs.items(): @@ -266,7 +265,7 @@ def _evaluate_obj(self): """ # NOTE: since we already have the attribute `obj`, # we can update it rather than setting it - if self.rtn.type != 'PF': + if self.rtn.obj is not None: self.rtn.obj.evaluate() self.obj = self.rtn.obj.optz @@ -342,7 +341,7 @@ def finalize(self, force=False): Returns True if the finalization is successful, False otherwise. """ # NOTE: for power flow type, we skip the finalization - if self.rtn.type == 'PF': + if self.rtn.class_name in ['PFlow']: self.finalized = True return self.finalized if self.finalized and not force: diff --git a/ams/routines/__init__.py b/ams/routines/__init__.py index 32df57cc..2d0dfd26 100644 --- a/ams/routines/__init__.py +++ b/ams/routines/__init__.py @@ -16,6 +16,7 @@ ('uc', ['UC', 'UCDG', 'UCES']), ('dopf', ['DOPF', 'DOPFVIS']), ('pflow0', ['PFlow0']), + ('dcpf2', ['DCPF2']), ]) class_names = list_flatten(list(all_routines.values())) From 2c9329f785aa468e2b5407fa9f0c6a9c930c4a70 Mon Sep 17 00:00:00 2001 From: jinningwang Date: Fri, 22 Nov 2024 16:20:26 -0500 Subject: [PATCH 69/96] Temp alleviate DCPF init --- ams/opt/omodel.py | 6 ++++-- 1 file changed, 4 insertions(+), 2 deletions(-) diff --git a/ams/opt/omodel.py b/ams/opt/omodel.py index 79f5f343..996daef0 100644 --- a/ams/opt/omodel.py +++ b/ams/opt/omodel.py @@ -209,7 +209,8 @@ def parse(self, force=False): self.rtn.obj.parse() except Exception as e: raise Exception(f"Failed to parse Objective <{self.rtn.obj.name}>.\n{e}") - elif not self.rtn.class_name in ['PFlow']: + # NOTE: after migrating to CVXPY DCPF, we should remove it from here + elif self.rtn.class_name not in ['PFlow', 'DCPF']: logger.warning(f"{self.rtn.class_name} has no objective function!") self.parsed = False return self.parsed @@ -341,7 +342,8 @@ def finalize(self, force=False): Returns True if the finalization is successful, False otherwise. """ # NOTE: for power flow type, we skip the finalization - if self.rtn.class_name in ['PFlow']: + # NOTE: after migrating to CVXPY DCPF, we should remove it from here + if self.rtn.class_name in ['PFlow', 'DCPF']: self.finalized = True return self.finalized if self.finalized and not force: From fccfb7f1e022db9080e94c991a439ede4651ea32 Mon Sep 17 00:00:00 2001 From: jinningwang Date: Fri, 22 Nov 2024 20:51:03 -0500 Subject: [PATCH 70/96] Typo --- ams/core/service.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/ams/core/service.py b/ams/core/service.py index 0f48653d..52015f9c 100644 --- a/ams/core/service.py +++ b/ams/core/service.py @@ -676,7 +676,7 @@ class VarSelect(NumOp): A numerical matrix to select a subset of a 2D variable, ``u.v[:, idx]``. - For example, if nned to select Energy Storage output + For example, if need to select Energy Storage output power from StaticGen `pg`, following definition can be used: ```python class RTED: From f218b6a1f2f701ca327104fcacb214bad55e439d Mon Sep 17 00:00:00 2001 From: jinningwang Date: Fri, 22 Nov 2024 20:51:35 -0500 Subject: [PATCH 71/96] Draft DCPF using CVXPY --- ams/routines/dcpf2.py | 177 +++++++++++++++++++++++++++++++++++++++--- 1 file changed, 166 insertions(+), 11 deletions(-) diff --git a/ams/routines/dcpf2.py b/ams/routines/dcpf2.py index fd67986c..0cda5a0d 100644 --- a/ams/routines/dcpf2.py +++ b/ams/routines/dcpf2.py @@ -3,21 +3,16 @@ """ import logging -from andes.shared import deg2rad -from andes.utils.misc import elapsed +from ams.opt import Var, Constraint, Expression, Objective +from ams.routines.routine import RoutineBase -from ams.opt import Var -from ams.pypower import runpf -from ams.pypower.core import ppoption -from ams.routines.dcopf import DCOPF - -from ams.io.pypower import system2ppc from ams.core.param import RParam +from ams.core.service import VarSelect logger = logging.getLogger(__name__) -class DCPF2(DCOPF): +class DCPF2(RoutineBase): """ DC power flow, overload the ``solve``, ``unpack``, and ``run`` methods. @@ -29,11 +24,171 @@ class DCPF2(DCOPF): """ def __init__(self, system, config): - DCOPF.__init__(self, system, config) + RoutineBase.__init__(self, system, config) self.info = 'DC Power Flow' self.type = 'PF' - self.obj.e_str = '0' + self.gsh = RParam(info='shunt conductance', + name='gsh', tex_name=r'g_{sh}', + model='Shunt', src='g', + no_parse=True,) + self.buss = RParam(info='Bus slack', + name='buss', tex_name=r'B_{us,s}', + model='Slack', src='bus', + no_parse=True,) + self.genpv = RParam(info='gen of PV', + name='genpv', tex_name=r'g_{DG}', + model='PV', src='idx', + no_parse=True,) + self.pd = RParam(info='active demand', + name='pd', tex_name=r'p_{d}', + model='StaticLoad', src='p0', + unit='p.u.',) + + # --- connection matrix --- + self.Cg = RParam(info='Gen connection matrix', + name='Cg', tex_name=r'C_{g}', + model='mats', src='Cg', + no_parse=True, sparse=True,) + self.Cl = RParam(info='Load connection matrix', + name='Cl', tex_name=r'C_{l}', + model='mats', src='Cl', + no_parse=True, sparse=True,) + self.CftT = RParam(info='Transpose of line connection matrix', + name='CftT', tex_name=r'C_{ft}^T', + model='mats', src='CftT', + no_parse=True, sparse=True,) + self.Csh = RParam(info='Shunt connection matrix', + name='Csh', tex_name=r'C_{sh}', + model='mats', src='Csh', + no_parse=True, sparse=True,) + + # --- system matrix --- + self.Bbus = RParam(info='Bus admittance matrix', + name='Bbus', tex_name=r'B_{bus}', + model='mats', src='Bbus', + no_parse=True, sparse=True,) + self.Bf = RParam(info='Bf matrix', + name='Bf', tex_name=r'B_{f}', + model='mats', src='Bf', + no_parse=True, sparse=True,) + self.Pbusinj = RParam(info='Bus power injection vector', + name='Pbusinj', tex_name=r'P_{bus}^{inj}', + model='mats', src='Pbusinj', + no_parse=True,) + self.Pfinj = RParam(info='Line power injection vector', + name='Pfinj', tex_name=r'P_{f}^{inj}', + model='mats', src='Pfinj', + no_parse=True,) + + self.pg = Var(info='Gen active power', + unit='p.u.', + name='pg', tex_name=r'p_g', + model='StaticGen', src='p') + self.aBus = Var(info='Bus voltage angle', + unit='rad', + name='aBus', tex_name=r'\theta_{bus}', + model='Bus', src='a',) + + self.cpv = VarSelect(u=self.pg, indexer='genpv', + name='cpv', tex_name=r'C_{ESD}', + info='Select PV from pg', + no_parse=True,) + self.pg0 = RParam(info='Gen initial active power', + name='pg0', tex_name=r'p_{g, 0}', + unit='p.u.', model='StaticGen', + src='p0', no_parse=False,) + + # --- power balance --- + pb = 'Bbus@aBus + Pbusinj + Cl@pd + Csh@gsh - Cg@pg' + self.pb = Constraint(name='pb', info='power balance', + e_str=pb, is_eq=True,) + self.pvb = Constraint(name='pvb', info='PV generator', + e_str='cpv * (pg - pg0)', + is_eq=True,) + + self.csb = VarSelect(info='select slack bus', + name='csb', tex_name=r'c_{sb}', + u=self.aBus, indexer='buss', + no_parse=True,) + self.sba = Constraint(info='align slack bus angle', + name='sbus', is_eq=True, + e_str='csb@aBus',) + self.plf = Expression(info='Line flow', + name='plf', tex_name=r'p_{lf}', + unit='p.u.', + e_str='Bf@aBus + Pfinj', + model='Line', src=None,) + + self.obj = Objective(name='obj', + info='total cost', unit='$', + sense='min', e_str='0',) + + def solve(self, **kwargs): + """ + Solve the routine optimization model. + args and kwargs go to `self.om.prob.solve()` (`cvxpy.Problem.solve()`). + """ + return self.om.prob.solve(**kwargs) + + def unpack(self, **kwargs): + """ + Unpack the results from CVXPY model. + """ + # --- solver Var results to routine algeb --- + for _, var in self.vars.items(): + # --- copy results from routine algeb into system algeb --- + if var.model is None: # if no owner + continue + if var.src is None: # if no source + continue + else: + try: + idx = var.owner.get_idx() + except AttributeError: + idx = var.owner.idx.v + else: + pass + # NOTE: only unpack the variables that are in the model or group + try: + var.owner.set(src=var.src, idx=idx, attr='v', value=var.v) + except (KeyError, TypeError): + logger.error(f'Failed to unpack <{var}> to <{var.owner.class_name}>.') + pass + + # --- solver ExpressionCalc results to routine algeb --- + for _, exprc in self.exprcs.items(): + if exprc.model is None: + continue + if exprc.src is None: + continue + else: + try: + idx = exprc.owner.get_idx() + except AttributeError: + idx = exprc.owner.idx.v + else: + pass + try: + exprc.owner.set(src=exprc.src, idx=idx, attr='v', value=exprc.v) + except (KeyError, TypeError): + logger.error(f'Failed to unpack <{exprc}> to <{exprc.owner.class_name}>.') + pass + + # label the most recent solved routine + self.system.recent = self.system.routines[self.class_name] + return True + + def _post_solve(self): + """ + Post-solve calculations. + """ + # NOTE: unpack Expressions if owner and arc are available + for expr in self.exprs.values(): + if expr.owner and expr.src: + expr.owner.set(src=expr.src, attr='v', + idx=expr.get_idx(), value=expr.v) + return True def summary(self, **kwargs): """ From 238bd76b3b13edc037caf4519a467f0b653cc2ef Mon Sep 17 00:00:00 2001 From: jinningwang Date: Fri, 22 Nov 2024 20:53:38 -0500 Subject: [PATCH 72/96] [WIP] In DCPF, switch from PYPOWER to CVXPY --- ams/routines/__init__.py | 2 +- ams/routines/acopf.py | 4 +- ams/routines/dcpf.py | 278 +++++++++++++++++++++------------------ ams/routines/dcpf0.py | 188 ++++++++++++++++++++++++++ ams/routines/dcpf2.py | 206 ----------------------------- ams/routines/pflow0.py | 4 +- 6 files changed, 341 insertions(+), 341 deletions(-) create mode 100644 ams/routines/dcpf0.py delete mode 100644 ams/routines/dcpf2.py diff --git a/ams/routines/__init__.py b/ams/routines/__init__.py index 2d0dfd26..ad5be239 100644 --- a/ams/routines/__init__.py +++ b/ams/routines/__init__.py @@ -16,7 +16,7 @@ ('uc', ['UC', 'UCDG', 'UCES']), ('dopf', ['DOPF', 'DOPFVIS']), ('pflow0', ['PFlow0']), - ('dcpf2', ['DCPF2']), + ('dcpf0', ['DCPF0']), ]) class_names = list_flatten(list(all_routines.values())) diff --git a/ams/routines/acopf.py b/ams/routines/acopf.py index b9e77528..0af6b87a 100644 --- a/ams/routines/acopf.py +++ b/ams/routines/acopf.py @@ -10,13 +10,13 @@ from ams.io.pypower import system2ppc from ams.core.param import RParam -from ams.routines.dcpf import DCPF +from ams.routines.dcpf0 import DCPF0 from ams.opt import Var, Constraint, Objective logger = logging.getLogger(__name__) -class ACOPF(DCPF): +class ACOPF(DCPF0): """ Standard AC optimal power flow. diff --git a/ams/routines/dcpf.py b/ams/routines/dcpf.py index ee7f46b6..acc70af3 100644 --- a/ams/routines/dcpf.py +++ b/ams/routines/dcpf.py @@ -3,16 +3,11 @@ """ import logging -from andes.shared import deg2rad -from andes.utils.misc import elapsed - +from ams.opt import Var, Constraint, Expression, Objective from ams.routines.routine import RoutineBase -from ams.opt import Var -from ams.pypower import runpf -from ams.pypower.core import ppoption -from ams.io.pypower import system2ppc from ams.core.param import RParam +from ams.core.service import VarSelect logger = logging.getLogger(__name__) @@ -33,147 +28,167 @@ def __init__(self, system, config): self.info = 'DC Power Flow' self.type = 'PF' - # --- routine data --- - self.x = RParam(info="line reactance", - name='x', tex_name='x', - unit='p.u.', - model='Line', src='x',) - self.tap = RParam(info="transformer branch tap ratio", - name='tap', tex_name=r't_{ap}', - model='Line', src='tap', - unit='float',) - self.phi = RParam(info="transformer branch phase shift in rad", - name='phi', tex_name=r'\phi', - model='Line', src='phi', - unit='radian',) - - # --- load --- - self.pd = RParam(info='active deman', + self.gsh = RParam(info='shunt conductance', + name='gsh', tex_name=r'g_{sh}', + model='Shunt', src='g', + no_parse=True,) + self.buss = RParam(info='Bus slack', + name='buss', tex_name=r'B_{us,s}', + model='Slack', src='bus', + no_parse=True,) + self.genpv = RParam(info='gen of PV', + name='genpv', tex_name=r'g_{DG}', + model='PV', src='idx', + no_parse=True,) + self.pd = RParam(info='active demand', name='pd', tex_name=r'p_{d}', - unit='p.u.', - model='StaticLoad', src='p0') - # --- gen --- + model='StaticLoad', src='p0', + unit='p.u.',) + + # --- connection matrix --- + self.Cg = RParam(info='Gen connection matrix', + name='Cg', tex_name=r'C_{g}', + model='mats', src='Cg', + no_parse=True, sparse=True,) + self.Cl = RParam(info='Load connection matrix', + name='Cl', tex_name=r'C_{l}', + model='mats', src='Cl', + no_parse=True, sparse=True,) + self.CftT = RParam(info='Transpose of line connection matrix', + name='CftT', tex_name=r'C_{ft}^T', + model='mats', src='CftT', + no_parse=True, sparse=True,) + self.Csh = RParam(info='Shunt connection matrix', + name='Csh', tex_name=r'C_{sh}', + model='mats', src='Csh', + no_parse=True, sparse=True,) + + # --- system matrix --- + self.Bbus = RParam(info='Bus admittance matrix', + name='Bbus', tex_name=r'B_{bus}', + model='mats', src='Bbus', + no_parse=True, sparse=True,) + self.Bf = RParam(info='Bf matrix', + name='Bf', tex_name=r'B_{f}', + model='mats', src='Bf', + no_parse=True, sparse=True,) + self.Pbusinj = RParam(info='Bus power injection vector', + name='Pbusinj', tex_name=r'P_{bus}^{inj}', + model='mats', src='Pbusinj', + no_parse=True,) + self.Pfinj = RParam(info='Line power injection vector', + name='Pfinj', tex_name=r'P_{f}^{inj}', + model='mats', src='Pfinj', + no_parse=True,) + self.pg = Var(info='Gen active power', unit='p.u.', - name='pg', tex_name=r'p_{g}', - model='StaticGen', src='p',) - - # --- bus --- - self.aBus = Var(info='bus voltage angle', + name='pg', tex_name=r'p_g', + model='StaticGen', src='p') + self.aBus = Var(info='Bus voltage angle', unit='rad', - name='aBus', tex_name=r'a_{Bus}', + name='aBus', tex_name=r'\theta_{bus}', model='Bus', src='a',) - # --- line flow --- - self.plf = Var(info='Line flow', - unit='p.u.', - name='plf', tex_name=r'p_{lf}', - model='Line',) - - def unpack(self, res): + self.cpv = VarSelect(u=self.pg, indexer='genpv', + name='cpv', tex_name=r'C_{ESD}', + info='Select PV from pg', + no_parse=True,) + self.pg0 = RParam(info='Gen initial active power', + name='pg0', tex_name=r'p_{g, 0}', + unit='p.u.', model='StaticGen', + src='p0', no_parse=False,) + + # --- power balance --- + pb = 'Bbus@aBus + Pbusinj + Cl@pd + Csh@gsh - Cg@pg' + self.pb = Constraint(name='pb', info='power balance', + e_str=pb, is_eq=True,) + self.pvb = Constraint(name='pvb', info='PV generator', + e_str='cpv * (pg - pg0)', + is_eq=True,) + + self.csb = VarSelect(info='select slack bus', + name='csb', tex_name=r'c_{sb}', + u=self.aBus, indexer='buss', + no_parse=True,) + self.sba = Constraint(info='align slack bus angle', + name='sbus', is_eq=True, + e_str='csb@aBus',) + self.plf = Expression(info='Line flow', + name='plf', tex_name=r'p_{lf}', + unit='p.u.', + e_str='Bf@aBus + Pfinj', + model='Line', src=None,) + + self.obj = Objective(name='obj', + info='total cost', unit='$', + sense='min', e_str='0',) + + def solve(self, **kwargs): """ - Unpack results from PYPOWER. + Solve the routine optimization model. + args and kwargs go to `self.om.prob.solve()` (`cvxpy.Problem.solve()`). """ - system = self.system - mva = res['baseMVA'] - - # --- copy results from ppc into system algeb --- - # --- Bus --- - system.Bus.v.v = res['bus'][:, 7] # voltage magnitude - system.Bus.a.v = res['bus'][:, 8] * deg2rad # voltage angle + return self.om.prob.solve(**kwargs) - # --- PV --- - system.PV.p.v = res['gen'][system.Slack.n:, 1] / mva # active power - system.PV.q.v = res['gen'][system.Slack.n:, 2] / mva # reactive power - - # --- Slack --- - system.Slack.p.v = res['gen'][:system.Slack.n, 1] / mva # active power - system.Slack.q.v = res['gen'][:system.Slack.n, 2] / mva # reactive power - - # --- Line --- - self.plf.optz.value = res['branch'][:, 13] / mva # line flow - - # --- copy results from system algeb into routine algeb --- - for vname, var in self.vars.items(): - owner = getattr(system, var.model) # instance of owner, Model or Group - if var.src is None: # skip if no source variable is specified + def unpack(self, **kwargs): + """ + Unpack the results from CVXPY model. + """ + # --- solver Var results to routine algeb --- + for _, var in self.vars.items(): + # --- copy results from routine algeb into system algeb --- + if var.model is None: # if no owner + continue + if var.src is None: # if no source continue - elif hasattr(owner, 'group'): # if owner is a Model instance - grp = getattr(system, owner.group) - idx = grp.get_idx() - elif hasattr(owner, 'get_idx'): # if owner is a Group instance - idx = owner.get_idx() else: - msg = f"Failed to find valid source variable `{owner.class_name}.{var.src}` for " - msg += f"{self.class_name}.{vname}, skip unpacking." - logger.warning(msg) + try: + idx = var.owner.get_idx() + except AttributeError: + idx = var.owner.idx.v + else: + pass + # NOTE: only unpack the variables that are in the model or group + try: + var.owner.set(src=var.src, idx=idx, attr='v', value=var.v) + except (KeyError, TypeError): + logger.error(f'Failed to unpack <{var}> to <{var.owner.class_name}>.') + pass + + # --- solver ExpressionCalc results to routine algeb --- + for _, exprc in self.exprcs.items(): + if exprc.model is None: continue - try: - logger.debug(f"Unpacking {vname} into {owner.class_name}.{var.src}.") - var.optz.value = owner.get(src=var.src, attr='v', idx=idx) - except AttributeError: - logger.debug(f"Failed to unpack {vname} into {owner.class_name}.{var.src}.") + if exprc.src is None: continue + else: + try: + idx = exprc.owner.get_idx() + except AttributeError: + idx = exprc.owner.idx.v + else: + pass + try: + exprc.owner.set(src=exprc.src, idx=idx, attr='v', value=exprc.v) + except (KeyError, TypeError): + logger.error(f'Failed to unpack <{exprc}> to <{exprc.owner.class_name}>.') + pass + + # label the most recent solved routine self.system.recent = self.system.routines[self.class_name] return True - def solve(self, method=None): - """ - Solve DC power flow using PYPOWER. + def _post_solve(self): """ - ppc = system2ppc(self.system) - ppopt = ppoption(PF_DC=True) - - res, sstats = runpf(casedata=ppc, ppopt=ppopt) - return res, sstats - - def run(self, **kwargs): + Post-solve calculations. """ - Run DC pwoer flow. - *args and **kwargs go to `self.solve()`, which are not used yet. - - Examples - -------- - >>> ss = ams.load(ams.get_case('matpower/case14.m')) - >>> ss.DCPF.run() - - Parameters - ---------- - method : str - Placeholder for future use. - - Returns - ------- - exit_code : int - Exit code of the routine. - """ - if not self.initialized: - self.init() - t0, _ = elapsed() - - res, sstats = self.solve(**kwargs) - self.converged = res['success'] - self.exit_code = 0 if res['success'] else 1 - _, s = elapsed(t0) - self.exec_time = float(s.split(' ')[0]) - n_iter = int(sstats['num_iters']) - n_iter_str = f"{n_iter} iterations " if n_iter > 1 else f"{n_iter} iteration " - if self.exit_code == 0: - msg = f"<{self.class_name}> solved in {s}, converged in " - msg += n_iter_str + f"with {sstats['solver_name']}." - logger.info(msg) - try: - self.unpack(res) - except Exception as e: - logger.error(f"Failed to unpack results from {self.class_name}.\n{e}") - return False - return True - else: - msg = f"{self.class_name} failed in " - msg += f"{int(sstats['num_iters'])} iterations with " - msg += f"{sstats['solver_name']}!" - logger.warning(msg) - return False + # NOTE: unpack Expressions if owner and arc are available + for expr in self.exprs.values(): + if expr.owner and expr.src: + expr.owner.set(src=expr.src, attr='v', + idx=expr.get_idx(), value=expr.v) + return True def summary(self, **kwargs): """ @@ -186,3 +201,6 @@ def enable(self, name): def disable(self, name): raise NotImplementedError + + def dc2ac(self, name): + raise NotImplementedError diff --git a/ams/routines/dcpf0.py b/ams/routines/dcpf0.py new file mode 100644 index 00000000..b7733a12 --- /dev/null +++ b/ams/routines/dcpf0.py @@ -0,0 +1,188 @@ +""" +Power flow routines. +""" +import logging + +from andes.shared import deg2rad +from andes.utils.misc import elapsed + +from ams.routines.routine import RoutineBase +from ams.opt import Var +from ams.pypower import runpf +from ams.pypower.core import ppoption + +from ams.io.pypower import system2ppc +from ams.core.param import RParam + +logger = logging.getLogger(__name__) + + +class DCPF0(RoutineBase): + """ + DC power flow, overload the ``solve``, ``unpack``, and ``run`` methods. + + Notes + ----- + 1. DCPF is solved with PYPOWER ``runpf`` function. + 2. DCPF formulation is not complete yet, but this does not affect the + results because the data are passed to PYPOWER for solving. + """ + + def __init__(self, system, config): + RoutineBase.__init__(self, system, config) + self.info = 'DC Power Flow' + self.type = 'PF' + + # --- routine data --- + self.x = RParam(info="line reactance", + name='x', tex_name='x', + unit='p.u.', + model='Line', src='x',) + self.tap = RParam(info="transformer branch tap ratio", + name='tap', tex_name=r't_{ap}', + model='Line', src='tap', + unit='float',) + self.phi = RParam(info="transformer branch phase shift in rad", + name='phi', tex_name=r'\phi', + model='Line', src='phi', + unit='radian',) + + # --- load --- + self.pd = RParam(info='active deman', + name='pd', tex_name=r'p_{d}', + unit='p.u.', + model='StaticLoad', src='p0') + # --- gen --- + self.pg = Var(info='Gen active power', + unit='p.u.', + name='pg', tex_name=r'p_{g}', + model='StaticGen', src='p',) + + # --- bus --- + self.aBus = Var(info='bus voltage angle', + unit='rad', + name='aBus', tex_name=r'a_{Bus}', + model='Bus', src='a',) + + # --- line flow --- + self.plf = Var(info='Line flow', + unit='p.u.', + name='plf', tex_name=r'p_{lf}', + model='Line',) + + def unpack(self, res): + """ + Unpack results from PYPOWER. + """ + system = self.system + mva = res['baseMVA'] + + # --- copy results from ppc into system algeb --- + # --- Bus --- + system.Bus.v.v = res['bus'][:, 7] # voltage magnitude + system.Bus.a.v = res['bus'][:, 8] * deg2rad # voltage angle + + # --- PV --- + system.PV.p.v = res['gen'][system.Slack.n:, 1] / mva # active power + system.PV.q.v = res['gen'][system.Slack.n:, 2] / mva # reactive power + + # --- Slack --- + system.Slack.p.v = res['gen'][:system.Slack.n, 1] / mva # active power + system.Slack.q.v = res['gen'][:system.Slack.n, 2] / mva # reactive power + + # --- Line --- + self.plf.optz.value = res['branch'][:, 13] / mva # line flow + + # --- copy results from system algeb into routine algeb --- + for vname, var in self.vars.items(): + owner = getattr(system, var.model) # instance of owner, Model or Group + if var.src is None: # skip if no source variable is specified + continue + elif hasattr(owner, 'group'): # if owner is a Model instance + grp = getattr(system, owner.group) + idx = grp.get_idx() + elif hasattr(owner, 'get_idx'): # if owner is a Group instance + idx = owner.get_idx() + else: + msg = f"Failed to find valid source variable `{owner.class_name}.{var.src}` for " + msg += f"{self.class_name}.{vname}, skip unpacking." + logger.warning(msg) + continue + try: + logger.debug(f"Unpacking {vname} into {owner.class_name}.{var.src}.") + var.optz.value = owner.get(src=var.src, attr='v', idx=idx) + except AttributeError: + logger.debug(f"Failed to unpack {vname} into {owner.class_name}.{var.src}.") + continue + self.system.recent = self.system.routines[self.class_name] + return True + + def solve(self, method=None): + """ + Solve DC power flow using PYPOWER. + """ + ppc = system2ppc(self.system) + ppopt = ppoption(PF_DC=True) + + res, sstats = runpf(casedata=ppc, ppopt=ppopt) + return res, sstats + + def run(self, **kwargs): + """ + Run DC pwoer flow. + *args and **kwargs go to `self.solve()`, which are not used yet. + + Examples + -------- + >>> ss = ams.load(ams.get_case('matpower/case14.m')) + >>> ss.DCPF.run() + + Parameters + ---------- + method : str + Placeholder for future use. + + Returns + ------- + exit_code : int + Exit code of the routine. + """ + if not self.initialized: + self.init() + t0, _ = elapsed() + + res, sstats = self.solve(**kwargs) + self.converged = res['success'] + self.exit_code = 0 if res['success'] else 1 + _, s = elapsed(t0) + self.exec_time = float(s.split(' ')[0]) + n_iter = int(sstats['num_iters']) + n_iter_str = f"{n_iter} iterations " if n_iter > 1 else f"{n_iter} iteration " + if self.exit_code == 0: + msg = f"<{self.class_name}> solved in {s}, converged in " + msg += n_iter_str + f"with {sstats['solver_name']}." + logger.info(msg) + try: + self.unpack(res) + except Exception as e: + logger.error(f"Failed to unpack results from {self.class_name}.\n{e}") + return False + return True + else: + msg = f"{self.class_name} failed in " + msg += f"{int(sstats['num_iters'])} iterations with " + msg += f"{sstats['solver_name']}!" + logger.warning(msg) + return False + + def summary(self, **kwargs): + """ + # TODO: Print power flow summary. + """ + raise NotImplementedError + + def enable(self, name): + raise NotImplementedError + + def disable(self, name): + raise NotImplementedError diff --git a/ams/routines/dcpf2.py b/ams/routines/dcpf2.py deleted file mode 100644 index 0cda5a0d..00000000 --- a/ams/routines/dcpf2.py +++ /dev/null @@ -1,206 +0,0 @@ -""" -Power flow routines. -""" -import logging - -from ams.opt import Var, Constraint, Expression, Objective -from ams.routines.routine import RoutineBase - -from ams.core.param import RParam -from ams.core.service import VarSelect - -logger = logging.getLogger(__name__) - - -class DCPF2(RoutineBase): - """ - DC power flow, overload the ``solve``, ``unpack``, and ``run`` methods. - - Notes - ----- - 1. DCPF is solved with PYPOWER ``runpf`` function. - 2. DCPF formulation is not complete yet, but this does not affect the - results because the data are passed to PYPOWER for solving. - """ - - def __init__(self, system, config): - RoutineBase.__init__(self, system, config) - self.info = 'DC Power Flow' - self.type = 'PF' - - self.gsh = RParam(info='shunt conductance', - name='gsh', tex_name=r'g_{sh}', - model='Shunt', src='g', - no_parse=True,) - self.buss = RParam(info='Bus slack', - name='buss', tex_name=r'B_{us,s}', - model='Slack', src='bus', - no_parse=True,) - self.genpv = RParam(info='gen of PV', - name='genpv', tex_name=r'g_{DG}', - model='PV', src='idx', - no_parse=True,) - self.pd = RParam(info='active demand', - name='pd', tex_name=r'p_{d}', - model='StaticLoad', src='p0', - unit='p.u.',) - - # --- connection matrix --- - self.Cg = RParam(info='Gen connection matrix', - name='Cg', tex_name=r'C_{g}', - model='mats', src='Cg', - no_parse=True, sparse=True,) - self.Cl = RParam(info='Load connection matrix', - name='Cl', tex_name=r'C_{l}', - model='mats', src='Cl', - no_parse=True, sparse=True,) - self.CftT = RParam(info='Transpose of line connection matrix', - name='CftT', tex_name=r'C_{ft}^T', - model='mats', src='CftT', - no_parse=True, sparse=True,) - self.Csh = RParam(info='Shunt connection matrix', - name='Csh', tex_name=r'C_{sh}', - model='mats', src='Csh', - no_parse=True, sparse=True,) - - # --- system matrix --- - self.Bbus = RParam(info='Bus admittance matrix', - name='Bbus', tex_name=r'B_{bus}', - model='mats', src='Bbus', - no_parse=True, sparse=True,) - self.Bf = RParam(info='Bf matrix', - name='Bf', tex_name=r'B_{f}', - model='mats', src='Bf', - no_parse=True, sparse=True,) - self.Pbusinj = RParam(info='Bus power injection vector', - name='Pbusinj', tex_name=r'P_{bus}^{inj}', - model='mats', src='Pbusinj', - no_parse=True,) - self.Pfinj = RParam(info='Line power injection vector', - name='Pfinj', tex_name=r'P_{f}^{inj}', - model='mats', src='Pfinj', - no_parse=True,) - - self.pg = Var(info='Gen active power', - unit='p.u.', - name='pg', tex_name=r'p_g', - model='StaticGen', src='p') - self.aBus = Var(info='Bus voltage angle', - unit='rad', - name='aBus', tex_name=r'\theta_{bus}', - model='Bus', src='a',) - - self.cpv = VarSelect(u=self.pg, indexer='genpv', - name='cpv', tex_name=r'C_{ESD}', - info='Select PV from pg', - no_parse=True,) - self.pg0 = RParam(info='Gen initial active power', - name='pg0', tex_name=r'p_{g, 0}', - unit='p.u.', model='StaticGen', - src='p0', no_parse=False,) - - # --- power balance --- - pb = 'Bbus@aBus + Pbusinj + Cl@pd + Csh@gsh - Cg@pg' - self.pb = Constraint(name='pb', info='power balance', - e_str=pb, is_eq=True,) - self.pvb = Constraint(name='pvb', info='PV generator', - e_str='cpv * (pg - pg0)', - is_eq=True,) - - self.csb = VarSelect(info='select slack bus', - name='csb', tex_name=r'c_{sb}', - u=self.aBus, indexer='buss', - no_parse=True,) - self.sba = Constraint(info='align slack bus angle', - name='sbus', is_eq=True, - e_str='csb@aBus',) - self.plf = Expression(info='Line flow', - name='plf', tex_name=r'p_{lf}', - unit='p.u.', - e_str='Bf@aBus + Pfinj', - model='Line', src=None,) - - self.obj = Objective(name='obj', - info='total cost', unit='$', - sense='min', e_str='0',) - - def solve(self, **kwargs): - """ - Solve the routine optimization model. - args and kwargs go to `self.om.prob.solve()` (`cvxpy.Problem.solve()`). - """ - return self.om.prob.solve(**kwargs) - - def unpack(self, **kwargs): - """ - Unpack the results from CVXPY model. - """ - # --- solver Var results to routine algeb --- - for _, var in self.vars.items(): - # --- copy results from routine algeb into system algeb --- - if var.model is None: # if no owner - continue - if var.src is None: # if no source - continue - else: - try: - idx = var.owner.get_idx() - except AttributeError: - idx = var.owner.idx.v - else: - pass - # NOTE: only unpack the variables that are in the model or group - try: - var.owner.set(src=var.src, idx=idx, attr='v', value=var.v) - except (KeyError, TypeError): - logger.error(f'Failed to unpack <{var}> to <{var.owner.class_name}>.') - pass - - # --- solver ExpressionCalc results to routine algeb --- - for _, exprc in self.exprcs.items(): - if exprc.model is None: - continue - if exprc.src is None: - continue - else: - try: - idx = exprc.owner.get_idx() - except AttributeError: - idx = exprc.owner.idx.v - else: - pass - try: - exprc.owner.set(src=exprc.src, idx=idx, attr='v', value=exprc.v) - except (KeyError, TypeError): - logger.error(f'Failed to unpack <{exprc}> to <{exprc.owner.class_name}>.') - pass - - # label the most recent solved routine - self.system.recent = self.system.routines[self.class_name] - return True - - def _post_solve(self): - """ - Post-solve calculations. - """ - # NOTE: unpack Expressions if owner and arc are available - for expr in self.exprs.values(): - if expr.owner and expr.src: - expr.owner.set(src=expr.src, attr='v', - idx=expr.get_idx(), value=expr.v) - return True - - def summary(self, **kwargs): - """ - # TODO: Print power flow summary. - """ - raise NotImplementedError - - def enable(self, name): - raise NotImplementedError - - def disable(self, name): - raise NotImplementedError - - def dc2ac(self, name): - raise NotImplementedError diff --git a/ams/routines/pflow0.py b/ams/routines/pflow0.py index 8dd72c15..b83c9c57 100644 --- a/ams/routines/pflow0.py +++ b/ams/routines/pflow0.py @@ -10,13 +10,13 @@ from ams.pypower.core import ppoption from ams.core.param import RParam -from ams.routines.dcpf import DCPF +from ams.routines.dcpf0 import DCPF0 from ams.opt import Var logger = logging.getLogger(__name__) -class PFlow0(DCPF): +class PFlow0(DCPF0): """ AC Power Flow routine. From 1f8a0844291955d789002a65794d5734b5993663 Mon Sep 17 00:00:00 2001 From: jinningwang Date: Fri, 22 Nov 2024 20:59:28 -0500 Subject: [PATCH 73/96] Typo --- ams/opt/omodel.py | 6 ++---- ams/routines/acopf.py | 2 +- ams/routines/dcpf.py | 2 +- ams/routines/pflow0.py | 2 +- 4 files changed, 5 insertions(+), 7 deletions(-) diff --git a/ams/opt/omodel.py b/ams/opt/omodel.py index 996daef0..7ae7a0d0 100644 --- a/ams/opt/omodel.py +++ b/ams/opt/omodel.py @@ -209,8 +209,7 @@ def parse(self, force=False): self.rtn.obj.parse() except Exception as e: raise Exception(f"Failed to parse Objective <{self.rtn.obj.name}>.\n{e}") - # NOTE: after migrating to CVXPY DCPF, we should remove it from here - elif self.rtn.class_name not in ['PFlow', 'DCPF']: + elif self.rtn.class_name not in ['PFlow', 'DCPF0']: logger.warning(f"{self.rtn.class_name} has no objective function!") self.parsed = False return self.parsed @@ -342,8 +341,7 @@ def finalize(self, force=False): Returns True if the finalization is successful, False otherwise. """ # NOTE: for power flow type, we skip the finalization - # NOTE: after migrating to CVXPY DCPF, we should remove it from here - if self.rtn.class_name in ['PFlow', 'DCPF']: + if self.rtn.class_name in ['PFlow', 'DCPF0']: self.finalized = True return self.finalized if self.finalized and not force: diff --git a/ams/routines/acopf.py b/ams/routines/acopf.py index 0af6b87a..e4403493 100644 --- a/ams/routines/acopf.py +++ b/ams/routines/acopf.py @@ -29,7 +29,7 @@ class ACOPF(DCPF0): """ def __init__(self, system, config): - DCPF.__init__(self, system, config) + DCPF0.__init__(self, system, config) self.info = 'AC Optimal Power Flow' self.type = 'ACED' diff --git a/ams/routines/dcpf.py b/ams/routines/dcpf.py index acc70af3..c44cae3b 100644 --- a/ams/routines/dcpf.py +++ b/ams/routines/dcpf.py @@ -104,7 +104,7 @@ def __init__(self, system, config): self.pb = Constraint(name='pb', info='power balance', e_str=pb, is_eq=True,) self.pvb = Constraint(name='pvb', info='PV generator', - e_str='cpv * (pg - pg0)', + e_str='cpv @ (pg - pg0)', is_eq=True,) self.csb = VarSelect(info='select slack bus', diff --git a/ams/routines/pflow0.py b/ams/routines/pflow0.py index b83c9c57..b2267b56 100644 --- a/ams/routines/pflow0.py +++ b/ams/routines/pflow0.py @@ -29,7 +29,7 @@ class PFlow0(DCPF0): """ def __init__(self, system, config): - DCPF.__init__(self, system, config) + DCPF0.__init__(self, system, config) self.info = "AC Power Flow" self.type = "PF" From 5778f2da441065f73dd81465b815364ee4f6a243 Mon Sep 17 00:00:00 2001 From: jinningwang Date: Fri, 22 Nov 2024 21:03:38 -0500 Subject: [PATCH 74/96] Typo --- ams/routines/dcpf.py | 8 ++++++-- 1 file changed, 6 insertions(+), 2 deletions(-) diff --git a/ams/routines/dcpf.py b/ams/routines/dcpf.py index c44cae3b..ae2975d5 100644 --- a/ams/routines/dcpf.py +++ b/ams/routines/dcpf.py @@ -28,6 +28,10 @@ def __init__(self, system, config): self.info = 'DC Power Flow' self.type = 'PF' + self.ug = RParam(info='Gen connection status', + name='ug', tex_name=r'u_{g}', + model='StaticGen', src='u', + no_parse=True) self.gsh = RParam(info='shunt conductance', name='gsh', tex_name=r'g_{sh}', model='Shunt', src='g', @@ -104,7 +108,7 @@ def __init__(self, system, config): self.pb = Constraint(name='pb', info='power balance', e_str=pb, is_eq=True,) self.pvb = Constraint(name='pvb', info='PV generator', - e_str='cpv @ (pg - pg0)', + e_str='cpv @ (pg - mul(ug, pg0))', is_eq=True,) self.csb = VarSelect(info='select slack bus', @@ -121,7 +125,7 @@ def __init__(self, system, config): model='Line', src=None,) self.obj = Objective(name='obj', - info='total cost', unit='$', + info='place holder', unit='$', sense='min', e_str='0',) def solve(self, **kwargs): From b597b4724e6468e86077bb635ee96ab1a400db0c Mon Sep 17 00:00:00 2001 From: jinningwang Date: Fri, 22 Nov 2024 21:09:48 -0500 Subject: [PATCH 75/96] Typo --- ams/routines/dcpf.py | 10 ++-------- 1 file changed, 2 insertions(+), 8 deletions(-) diff --git a/ams/routines/dcpf.py b/ams/routines/dcpf.py index ae2975d5..cd7abeb4 100644 --- a/ams/routines/dcpf.py +++ b/ams/routines/dcpf.py @@ -14,13 +14,7 @@ class DCPF(RoutineBase): """ - DC power flow, overload the ``solve``, ``unpack``, and ``run`` methods. - - Notes - ----- - 1. DCPF is solved with PYPOWER ``runpf`` function. - 2. DCPF formulation is not complete yet, but this does not affect the - results because the data are passed to PYPOWER for solving. + DC power flow. """ def __init__(self, system, config): @@ -95,7 +89,7 @@ def __init__(self, system, config): model='Bus', src='a',) self.cpv = VarSelect(u=self.pg, indexer='genpv', - name='cpv', tex_name=r'C_{ESD}', + name='cpv', tex_name=r'C_{PV}', info='Select PV from pg', no_parse=True,) self.pg0 = RParam(info='Gen initial active power', From d3ce6a714b919e8a16f49a901cb616fa1ebcc6ce Mon Sep 17 00:00:00 2001 From: jinningwang Date: Fri, 22 Nov 2024 21:13:20 -0500 Subject: [PATCH 76/96] Add place holder obj for PFlow routine --- ams/opt/omodel.py | 4 ++-- ams/routines/pflow.py | 6 +++++- 2 files changed, 7 insertions(+), 3 deletions(-) diff --git a/ams/opt/omodel.py b/ams/opt/omodel.py index 7ae7a0d0..8c254aa6 100644 --- a/ams/opt/omodel.py +++ b/ams/opt/omodel.py @@ -209,7 +209,7 @@ def parse(self, force=False): self.rtn.obj.parse() except Exception as e: raise Exception(f"Failed to parse Objective <{self.rtn.obj.name}>.\n{e}") - elif self.rtn.class_name not in ['PFlow', 'DCPF0']: + elif self.rtn.class_name not in ['DCPF0']: logger.warning(f"{self.rtn.class_name} has no objective function!") self.parsed = False return self.parsed @@ -341,7 +341,7 @@ def finalize(self, force=False): Returns True if the finalization is successful, False otherwise. """ # NOTE: for power flow type, we skip the finalization - if self.rtn.class_name in ['PFlow', 'DCPF0']: + if self.rtn.class_name in ['DCPF0']: self.finalized = True return self.finalized if self.finalized and not force: diff --git a/ams/routines/pflow.py b/ams/routines/pflow.py index b029d5c0..d2a31357 100644 --- a/ams/routines/pflow.py +++ b/ams/routines/pflow.py @@ -8,7 +8,7 @@ from ams.core.param import RParam from ams.routines.routine import RoutineBase -from ams.opt import Var, Expression +from ams.opt import Var, Expression, Objective from ams.interface import _to_andes_pflow, sync_adsys logger = logging.getLogger(__name__) @@ -92,6 +92,10 @@ def __init__(self, system, config): e_str='Bf@aBus + Pfinj', model='Line', src=None,) + self.obj = Objective(name='obj', + info='place holder', unit='$', + sense='min', e_str='0',) + def init(self, **kwargs): """ Initialize the ANDES PFlow routine. From ade36cba71e0dc38dc4ca9498cd93e73862323be Mon Sep 17 00:00:00 2001 From: jinningwang Date: Fri, 22 Nov 2024 21:16:25 -0500 Subject: [PATCH 77/96] Add deprecation notice in DCPF0 and PFlow0 --- ams/routines/dcpf0.py | 4 +++- ams/routines/pflow0.py | 4 +++- 2 files changed, 6 insertions(+), 2 deletions(-) diff --git a/ams/routines/dcpf0.py b/ams/routines/dcpf0.py index b7733a12..fae566e4 100644 --- a/ams/routines/dcpf0.py +++ b/ams/routines/dcpf0.py @@ -19,7 +19,9 @@ class DCPF0(RoutineBase): """ - DC power flow, overload the ``solve``, ``unpack``, and ``run`` methods. + DC power flow using PYPOWER. + + This class is deprecated as of version 0.9.12 and will be removed in 1.0.0. Notes ----- diff --git a/ams/routines/pflow0.py b/ams/routines/pflow0.py index b2267b56..7e05fd6e 100644 --- a/ams/routines/pflow0.py +++ b/ams/routines/pflow0.py @@ -18,7 +18,9 @@ class PFlow0(DCPF0): """ - AC Power Flow routine. + AC Power Flow using PYPOWER. + + This class is deprecated as of version 0.9.12 and will be removed in 1.0.0. Notes ----- From 94c81ac50caf9ad3d2bbbd4ed89d0c92c5b31264 Mon Sep 17 00:00:00 2001 From: jinningwang Date: Fri, 22 Nov 2024 21:19:47 -0500 Subject: [PATCH 78/96] In Documenter, set Srouce to be owner if there is no src --- ams/core/documenter.py | 2 ++ 1 file changed, 2 insertions(+) diff --git a/ams/core/documenter.py b/ams/core/documenter.py index 22bab272..4f694b03 100644 --- a/ams/core/documenter.py +++ b/ams/core/documenter.py @@ -334,6 +334,8 @@ def _expr_doc(self, max_width=78, export='plain'): slist = [] if p.owner is not None and p.src is not None: slist.append(f'{p.owner.class_name}.{p.src}') + elif p.owner is not None and p.src is None: + slist.append(f'{p.owner.class_name}') sources.append(','.join(slist)) # expressions based on output format From ff8020048b1e97cb4ba4282da34d4c9a241d1e03 Mon Sep 17 00:00:00 2001 From: jinningwang Date: Fri, 22 Nov 2024 21:21:23 -0500 Subject: [PATCH 79/96] In Documenter, set Srouce to be owner if there is no src --- ams/core/documenter.py | 6 ++++++ ams/routines/dcopf.py | 4 ++-- 2 files changed, 8 insertions(+), 2 deletions(-) diff --git a/ams/core/documenter.py b/ams/core/documenter.py index 4f694b03..21b85501 100644 --- a/ams/core/documenter.py +++ b/ams/core/documenter.py @@ -387,6 +387,8 @@ def _exprc_doc(self, max_width=78, export='plain'): slist = [] if p.owner is not None and p.src is not None: slist.append(f'{p.owner.class_name}.{p.src}') + elif p.owner is not None and p.src is None: + slist.append(f'{p.owner.class_name}') sources.append(','.join(slist)) # expressions based on output format @@ -485,6 +487,8 @@ def _var_doc(self, max_width=78, export='plain'): slist = [] if p.owner is not None and p.src is not None: slist.append(f'{p.owner.class_name}.{p.src}') + elif p.owner is not None and p.src is None: + slist.append(f'{p.owner.class_name}') sources.append(','.join(slist)) # symbols based on output format @@ -552,6 +556,8 @@ def _param_doc(self, max_width=78, export='plain'): slist = [] if p.owner is not None and p.src is not None: slist.append(f'{p.owner.class_name}.{p.src}') + elif p.owner is not None and p.src is None: + slist.append(f'{p.owner.class_name}') sources.append(','.join(slist)) # symbols based on output format diff --git a/ams/routines/dcopf.py b/ams/routines/dcopf.py index 11644fe8..825f8ce7 100644 --- a/ams/routines/dcopf.py +++ b/ams/routines/dcopf.py @@ -171,11 +171,11 @@ def __init__(self, system, config): self.pmaxe = Expression(info='Effective pmax', name='pmaxe', tex_name=r'p_{g, max, e}', e_str='mul(nctrle, pg0) + mul(ctrle, pmax)', - model='StaticGen', src=None,) + model='StaticGen', src=None, unit='p.u.',) self.pmine = Expression(info='Effective pmin', name='pmine', tex_name=r'p_{g, min, e}', e_str='mul(nctrle, pg0) + mul(ctrle, pmin)', - model='StaticGen', src=None,) + model='StaticGen', src=None, unit='p.u.',) self.pglb = Constraint(name='pglb', info='pg min', e_str='-pg + pmine', is_eq=False,) self.pgub = Constraint(name='pgub', info='pg max', From bfea3d8ce292934dc3da86d3e8f4cad199bd47bb Mon Sep 17 00:00:00 2001 From: jinningwang Date: Fri, 22 Nov 2024 21:27:28 -0500 Subject: [PATCH 80/96] Typo --- ams/routines/pflow.py | 12 ++++-------- ams/routines/rted.py | 2 +- 2 files changed, 5 insertions(+), 9 deletions(-) diff --git a/ams/routines/pflow.py b/ams/routines/pflow.py index d2a31357..a769c91f 100644 --- a/ams/routines/pflow.py +++ b/ams/routines/pflow.py @@ -18,20 +18,16 @@ class PFlow(RoutineBase): """ Power flow analysis using ANDES PFlow routine. - More settings can be changed via `PFlow2._adsys.config` and `PFlow2._adsys.PFlow.config`. + More settings can be changed via ``PFlow2._adsys.config`` and ``PFlow2._adsys.PFlow.config``. All generator output powers, bus voltages, and angles are included in the variable definitions. However, not all of these are unknowns; the definitions are provided for easy access. - Reference - --------- - + References + ---------- [1] M. L. Crow, Computational methods for electric power systems. 2015. - [2] ANDES Documentation - Simulation and Plot. [Online]. - - Available: - https://docs.andes.app/en/latest/_examples/ex1.html + [2] ANDES Documentation - Simulation and Plot. [Online]. Available: https://docs.andes.app/en/latest/_examples/ex1.html """ def __init__(self, system, config): diff --git a/ams/routines/rted.py b/ams/routines/rted.py index 36ecccf8..66aa1fe9 100644 --- a/ams/routines/rted.py +++ b/ams/routines/rted.py @@ -303,7 +303,7 @@ def __init__(self): name='gendg', tex_name=r'g_{DG}', model='DG', src='gen', no_parse=True,) - info = 'Ratio of DG.pge w.r.t to that of static generator', + info = 'Ratio of DG.pge w.r.t to that of static generator' self.gammapdg = RParam(name='gammapd', tex_name=r'\gamma_{p,DG}', model='DG', src='gammap', no_parse=True, info=info) From bdb340152292625fd15c2dca8af4f36fb3a0cbd1 Mon Sep 17 00:00:00 2001 From: jinningwang Date: Fri, 22 Nov 2024 21:42:40 -0500 Subject: [PATCH 81/96] Format --- ams/routines/pflow.py | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/ams/routines/pflow.py b/ams/routines/pflow.py index a769c91f..34d6c145 100644 --- a/ams/routines/pflow.py +++ b/ams/routines/pflow.py @@ -27,7 +27,8 @@ class PFlow(RoutineBase): ---------- [1] M. L. Crow, Computational methods for electric power systems. 2015. - [2] ANDES Documentation - Simulation and Plot. [Online]. Available: https://docs.andes.app/en/latest/_examples/ex1.html + [2] ANDES Documentation - Simulation and Plot. [Online]. Available: + https://docs.andes.app/en/latest/_examples/ex1.html """ def __init__(self, system, config): From 0fde998e8d98bce957f543449eb10d9d4804a2ce Mon Sep 17 00:00:00 2001 From: jinningwang Date: Fri, 22 Nov 2024 21:46:04 -0500 Subject: [PATCH 82/96] Refactor DCPF and DCOPF --- ams/routines/dcopf.py | 162 ++------------------------------------- ams/routines/dcpf.py | 77 ++++++++++--------- tests/test_known_good.py | 6 +- 3 files changed, 52 insertions(+), 193 deletions(-) diff --git a/ams/routines/dcopf.py b/ams/routines/dcopf.py index 825f8ce7..1fb0ed5f 100644 --- a/ams/routines/dcopf.py +++ b/ams/routines/dcopf.py @@ -5,17 +5,17 @@ import numpy as np from ams.core.param import RParam -from ams.core.service import NumOp, NumOpDual, VarSelect +from ams.core.service import NumOp, NumOpDual -from ams.routines.routine import RoutineBase +from ams.routines.dcpf import DCPFBase -from ams.opt import Var, Constraint, Objective, ExpressionCalc, Expression +from ams.opt import Constraint, Objective, ExpressionCalc, Expression logger = logging.getLogger(__name__) -class DCOPF(RoutineBase): +class DCOPF(DCPFBase): """ DC optimal power flow (DCOPF). @@ -29,7 +29,7 @@ class DCOPF(RoutineBase): """ def __init__(self, system, config): - RoutineBase.__init__(self, system, config) + DCPFBase.__init__(self, system, config) self.info = 'DC Optimal Power Flow' self.type = 'DCED' @@ -62,10 +62,6 @@ def __init__(self, system, config): indexer='gen', imodel='StaticGen', no_parse=True) # --- generator --- - self.ug = RParam(info='Gen connection status', - name='ug', tex_name=r'u_{g}', - model='StaticGen', src='u', - no_parse=True) self.ctrl = RParam(info='Gen controllability', name='ctrl', tex_name=r'c_{trl}', model='StaticGen', src='ctrl', @@ -90,20 +86,7 @@ def __init__(self, system, config): name='pmin', tex_name=r'p_{g, min}', unit='p.u.', model='StaticGen', no_parse=False,) - self.pg0 = RParam(info='Gen initial active power', - name='pg0', tex_name=r'p_{g, 0}', - unit='p.u.', model='StaticGen', - src='p0', no_parse=False,) - # --- bus --- - self.buss = RParam(info='Bus slack', - name='buss', tex_name=r'B_{us,s}', - model='Slack', src='bus', - no_parse=True,) - # --- load --- - self.pd = RParam(info='active demand', - name='pd', tex_name=r'p_{d}', - model='StaticLoad', src='p0', - unit='p.u.',) + # --- line --- self.ul = RParam(info='Line connection status', name='ul', tex_name=r'u_{l}', @@ -121,53 +104,8 @@ def __init__(self, system, config): name='amin', tex_name=r'\theta_{bus, min}', info='min line angle difference', no_parse=True,) - # --- shunt --- - self.gsh = RParam(info='shunt conductance', - name='gsh', tex_name=r'g_{sh}', - model='Shunt', src='g', - no_parse=True,) - # --- connection matrix --- - self.Cg = RParam(info='Gen connection matrix', - name='Cg', tex_name=r'C_{g}', - model='mats', src='Cg', - no_parse=True, sparse=True,) - self.Cl = RParam(info='Load connection matrix', - name='Cl', tex_name=r'C_{l}', - model='mats', src='Cl', - no_parse=True, sparse=True,) - self.CftT = RParam(info='Transpose of line connection matrix', - name='CftT', tex_name=r'C_{ft}^T', - model='mats', src='CftT', - no_parse=True, sparse=True,) - self.Csh = RParam(info='Shunt connection matrix', - name='Csh', tex_name=r'C_{sh}', - model='mats', src='Csh', - no_parse=True, sparse=True,) - # --- system matrix --- - self.Bbus = RParam(info='Bus admittance matrix', - name='Bbus', tex_name=r'B_{bus}', - model='mats', src='Bbus', - no_parse=True, sparse=True,) - self.Bf = RParam(info='Bf matrix', - name='Bf', tex_name=r'B_{f}', - model='mats', src='Bf', - no_parse=True, sparse=True,) - self.Pbusinj = RParam(info='Bus power injection vector', - name='Pbusinj', tex_name=r'P_{bus}^{inj}', - model='mats', src='Pbusinj', - no_parse=True,) - self.Pfinj = RParam(info='Line power injection vector', - name='Pfinj', tex_name=r'P_{f}^{inj}', - model='mats', src='Pfinj', - no_parse=True,) # --- Model Section --- - # --- generation --- - self.pg = Var(info='Gen active power', - unit='p.u.', - name='pg', tex_name=r'p_g', - model='StaticGen', src='p', - v0=self.pg0) self.pmaxe = Expression(info='Effective pmax', name='pmaxe', tex_name=r'p_{g, max, e}', e_str='mul(nctrle, pg0) + mul(ctrle, pmax)', @@ -180,28 +118,8 @@ def __init__(self, system, config): e_str='-pg + pmine', is_eq=False,) self.pgub = Constraint(name='pgub', info='pg max', e_str='pg - pmaxe', is_eq=False,) - # --- bus --- - self.aBus = Var(info='Bus voltage angle', - unit='rad', - name='aBus', tex_name=r'\theta_{bus}', - model='Bus', src='a',) - self.csb = VarSelect(info='select slack bus', - name='csb', tex_name=r'c_{sb}', - u=self.aBus, indexer='buss', - no_parse=True,) - self.sba = Constraint(info='align slack bus angle', - name='sbus', is_eq=True, - e_str='csb@aBus',) - # --- power balance --- - pb = 'Bbus@aBus + Pbusinj + Cl@pd + Csh@gsh - Cg@pg' - self.pb = Constraint(name='pb', info='power balance', - e_str=pb, is_eq=True,) + # --- line flow --- - self.plf = Expression(info='Line flow', - name='plf', tex_name=r'p_{lf}', - unit='p.u.', - e_str='Bf@aBus + Pfinj', - model='Line', src=None,) self.plflb = Constraint(info='line flow lower bound', name='plflb', is_eq=False, e_str='-plf - mul(ul, rate_a)',) @@ -228,13 +146,6 @@ def __init__(self, system, config): info='total cost', unit='$', sense='min', e_str=obj,) - def solve(self, **kwargs): - """ - Solve the routine optimization model. - args and kwargs go to `self.om.prob.solve()` (`cvxpy.Problem.solve()`). - """ - return self.om.prob.solve(**kwargs) - def run(self, **kwargs): """ Run the routine. @@ -282,65 +193,6 @@ def run(self, **kwargs): """ return super().run(**kwargs) - def _post_solve(self): - """ - Post-solve calculations. - """ - # NOTE: unpack Expressions if owner and arc are available - for expr in self.exprs.values(): - if expr.owner and expr.src: - expr.owner.set(src=expr.src, attr='v', - idx=expr.get_idx(), value=expr.v) - return True - - def unpack(self, **kwargs): - """ - Unpack the results from CVXPY model. - """ - # --- solver Var results to routine algeb --- - for _, var in self.vars.items(): - # --- copy results from routine algeb into system algeb --- - if var.model is None: # if no owner - continue - if var.src is None: # if no source - continue - else: - try: - idx = var.owner.get_idx() - except AttributeError: - idx = var.owner.idx.v - else: - pass - # NOTE: only unpack the variables that are in the model or group - try: - var.owner.set(src=var.src, idx=idx, attr='v', value=var.v) - except (KeyError, TypeError): - logger.error(f'Failed to unpack <{var}> to <{var.owner.class_name}>.') - pass - - # --- solver ExpressionCalc results to routine algeb --- - for _, exprc in self.exprcs.items(): - if exprc.model is None: - continue - if exprc.src is None: - continue - else: - try: - idx = exprc.owner.get_idx() - except AttributeError: - idx = exprc.owner.idx.v - else: - pass - try: - exprc.owner.set(src=exprc.src, idx=idx, attr='v', value=exprc.v) - except (KeyError, TypeError): - logger.error(f'Failed to unpack <{exprc}> to <{exprc.owner.class_name}>.') - pass - - # label the most recent solved routine - self.system.recent = self.system.routines[self.class_name] - return True - def dc2ac(self, kloss=1.0, **kwargs): """ Convert the DCOPF results with ACOPF. diff --git a/ams/routines/dcpf.py b/ams/routines/dcpf.py index cd7abeb4..e99a59ca 100644 --- a/ams/routines/dcpf.py +++ b/ams/routines/dcpf.py @@ -12,32 +12,33 @@ logger = logging.getLogger(__name__) -class DCPF(RoutineBase): +class DCPFBase(RoutineBase): """ - DC power flow. + Base class for DC power flow. """ def __init__(self, system, config): RoutineBase.__init__(self, system, config) - self.info = 'DC Power Flow' - self.type = 'PF' self.ug = RParam(info='Gen connection status', name='ug', tex_name=r'u_{g}', model='StaticGen', src='u', no_parse=True) + self.pg0 = RParam(info='Gen initial active power', + name='pg0', tex_name=r'p_{g, 0}', + unit='p.u.', model='StaticGen', + src='p0', no_parse=False,) + # --- shunt --- self.gsh = RParam(info='shunt conductance', name='gsh', tex_name=r'g_{sh}', model='Shunt', src='g', no_parse=True,) + self.buss = RParam(info='Bus slack', name='buss', tex_name=r'B_{us,s}', model='Slack', src='bus', no_parse=True,) - self.genpv = RParam(info='gen of PV', - name='genpv', tex_name=r'g_{DG}', - model='PV', src='idx', - no_parse=True,) + # --- load --- self.pd = RParam(info='active demand', name='pd', tex_name=r'p_{d}', model='StaticLoad', src='p0', @@ -79,32 +80,25 @@ def __init__(self, system, config): model='mats', src='Pfinj', no_parse=True,) + # --- generation --- self.pg = Var(info='Gen active power', unit='p.u.', name='pg', tex_name=r'p_g', - model='StaticGen', src='p') + model='StaticGen', src='p', + v0=self.pg0) + + # --- bus --- self.aBus = Var(info='Bus voltage angle', unit='rad', name='aBus', tex_name=r'\theta_{bus}', model='Bus', src='a',) - self.cpv = VarSelect(u=self.pg, indexer='genpv', - name='cpv', tex_name=r'C_{PV}', - info='Select PV from pg', - no_parse=True,) - self.pg0 = RParam(info='Gen initial active power', - name='pg0', tex_name=r'p_{g, 0}', - unit='p.u.', model='StaticGen', - src='p0', no_parse=False,) - # --- power balance --- pb = 'Bbus@aBus + Pbusinj + Cl@pd + Csh@gsh - Cg@pg' self.pb = Constraint(name='pb', info='power balance', e_str=pb, is_eq=True,) - self.pvb = Constraint(name='pvb', info='PV generator', - e_str='cpv @ (pg - mul(ug, pg0))', - is_eq=True,) + # --- bus --- self.csb = VarSelect(info='select slack bus', name='csb', tex_name=r'c_{sb}', u=self.aBus, indexer='buss', @@ -112,16 +106,14 @@ def __init__(self, system, config): self.sba = Constraint(info='align slack bus angle', name='sbus', is_eq=True, e_str='csb@aBus',) + + # --- line flow --- self.plf = Expression(info='Line flow', name='plf', tex_name=r'p_{lf}', unit='p.u.', e_str='Bf@aBus + Pfinj', model='Line', src=None,) - self.obj = Objective(name='obj', - info='place holder', unit='$', - sense='min', e_str='0',) - def solve(self, **kwargs): """ Solve the routine optimization model. @@ -188,17 +180,30 @@ def _post_solve(self): idx=expr.get_idx(), value=expr.v) return True - def summary(self, **kwargs): - """ - # TODO: Print power flow summary. - """ - raise NotImplementedError - def enable(self, name): - raise NotImplementedError +class DCPF(DCPFBase): + """ + DC power flow. + """ - def disable(self, name): - raise NotImplementedError + def __init__(self, system, config): + DCPFBase.__init__(self, system, config) + self.info = 'DC Power Flow' + self.type = 'PF' - def dc2ac(self, name): - raise NotImplementedError + self.genpv = RParam(info='gen of PV', + name='genpv', tex_name=r'g_{DG}', + model='PV', src='idx', + no_parse=True,) + self.cpv = VarSelect(u=self.pg, indexer='genpv', + name='cpv', tex_name=r'C_{PV}', + info='Select PV from pg', + no_parse=True,) + + self.pvb = Constraint(name='pvb', info='PV generator', + e_str='cpv @ (pg - mul(ug, pg0))', + is_eq=True,) + + self.obj = Objective(name='obj', + info='place holder', unit='$', + sense='min', e_str='0',) diff --git a/tests/test_known_good.py b/tests/test_known_good.py index 7f72d894..e53d1839 100644 --- a/tests/test_known_good.py +++ b/tests/test_known_good.py @@ -164,8 +164,10 @@ def test_DCPF_case118(self): Test DC power flow for case118. """ self.sp.DCPF.run() - np.testing.assert_allclose(self.sp.DCPF.aBus.v * rad2deg, - np.array(self.mpres['case118']['DCPF']['aBus']).reshape(-1), + aBus_mp = np.array(self.mpres['case118']['DCPF']['aBus']).reshape(-1) + aBus_mp -= aBus_mp[0] + np.testing.assert_allclose((self.sp.DCPF.aBus.v - self.sp.DCPF.aBus.v[0]) * rad2deg, + aBus_mp, rtol=1e-2, atol=1e-2) np.testing.assert_allclose(self.sp.DCPF.pg.v.sum() * self.sp.config.mva, From 6635382b176da0e8e391b199c7e98a4d8aeafb3c Mon Sep 17 00:00:00 2001 From: jinningwang Date: Fri, 22 Nov 2024 21:59:14 -0500 Subject: [PATCH 83/96] Typo --- ams/pypower/routines/opf.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/ams/pypower/routines/opf.py b/ams/pypower/routines/opf.py index a58c4e37..a855dcf6 100644 --- a/ams/pypower/routines/opf.py +++ b/ams/pypower/routines/opf.py @@ -1203,7 +1203,7 @@ def linear_constraints(self): # nnzA = nnzA + nnz(self.lin["data"].A.(self.lin.order{k})) if self.lin["N"]: - A = sp.sparse.lil_matrix((self.lin["N"], self.var["N"])) + A = sp.lil_matrix((self.lin["N"], self.var["N"])) u = inf * np.ones(self.lin["N"]) l = -u else: From 15611e502be282b830b5aff8fa111422a2cc83e8 Mon Sep 17 00:00:00 2001 From: jinningwang Date: Fri, 22 Nov 2024 22:56:18 -0500 Subject: [PATCH 84/96] Typo --- ams/routines/dcpf0.py | 2 +- ams/routines/pflow0.py | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/ams/routines/dcpf0.py b/ams/routines/dcpf0.py index fae566e4..d3a7403c 100644 --- a/ams/routines/dcpf0.py +++ b/ams/routines/dcpf0.py @@ -21,7 +21,7 @@ class DCPF0(RoutineBase): """ DC power flow using PYPOWER. - This class is deprecated as of version 0.9.12 and will be removed in 1.0.0. + This class is deprecated as of version 0.9.12 and will be removed in 1.1.0. Notes ----- diff --git a/ams/routines/pflow0.py b/ams/routines/pflow0.py index 7e05fd6e..74f079fc 100644 --- a/ams/routines/pflow0.py +++ b/ams/routines/pflow0.py @@ -20,7 +20,7 @@ class PFlow0(DCPF0): """ AC Power Flow using PYPOWER. - This class is deprecated as of version 0.9.12 and will be removed in 1.0.0. + This class is deprecated as of version 0.9.12 and will be removed in 1.1.0. Notes ----- From 4b546289193d23538b3624ca6b7b61069ccad006 Mon Sep 17 00:00:00 2001 From: jinningwang Date: Sat, 23 Nov 2024 12:31:11 -0500 Subject: [PATCH 85/96] Update release notes --- docs/source/release-notes.rst | 71 +++++++++++++++++++---------------- 1 file changed, 39 insertions(+), 32 deletions(-) diff --git a/docs/source/release-notes.rst b/docs/source/release-notes.rst index 8bc5788a..f0c4f0f0 100644 --- a/docs/source/release-notes.rst +++ b/docs/source/release-notes.rst @@ -12,12 +12,16 @@ Pre-v1.0.0 v0.9.12 (202x-xx-xx) -------------------- -- Refactor `OModel.initialized` as a property method -- Add a demo to show using `Constraint.e` for debugging -- Fix `ams.opt.omodel.Param.evaluate()` when its value is a number -- Improve `ams.opt.omodel.ExpressionCalc()` for efficiency -- Refactor module `ams.opt` -- Add class `ams.opt.Expression` +- Refactor ``OModel.initialized`` as a property method +- Add a demo to show using ``Constraint.e`` for debugging +- Fix ``ams.opt.omodel.Param.evaluate()`` when its value is a number +- Improve ``ams.opt.omodel.ExpressionCalc()`` for better performance +- Refactor module ``ams.opt`` +- Add class ``ams.opt.Expression`` +- Switch from PYPOWER to ANDES in routine ``PFlow`` +- Switch from PYPOWER to regular formulation in routine ``DCPF`` +- Refactor routines ``DCPF`` and ``DCOPF`` +- In ``RDocumenter``, set Srouce to be owner if there is no src RC1 ~~~~ @@ -27,34 +31,35 @@ v0.9.11 (2024-11-14) -------------------- - Add pyproject.toml for PEP 517 and PEP 518 compliance -- Add model `Jumper` -- Fix deprecation warning related to `pandas.fillna` and `newshape` in NumPy -- Minor refactor on solvers information in the module `shared` +- Add model ``Jumper`` +- Fix deprecation warning related to ``pandas.fillna`` and ``newshape`` in NumPy +- Minor refactor on solvers information in the module ``shared`` - Change default values of minimum ON/OFF duration time of generators to be 1 and 0.5 hours -- Add parameter `uf` for enforced generator on/off status -- In servicee `LoadScale`, consider load online status -- Consider line online status in routine `ED` -- Add methods `evaluate` and `finalize` in the class `OModel` to handle optimization elements generation and assembling -- Refactor `OModel.init()` and `Routine.init()` +- Add parameter ``uf`` for enforced generator on/off status +- In servicee ``LoadScale``, consider load online status +- Consider line online status in routine ``ED`` +- Add methods ``evaluate`` and ``finalize`` in the class ``OModel`` to handle optimization + elements generation and assembling +- Refactor ``OModel.init()`` and ``Routine.init()`` - Add ANDES paper as a citation file for now - Add more routine tests for generator trip, line trip, and load trip - Add a README to overview built-in cases -- Rename methods `v2` as `e` for classes `Constraint` and `Objective` +- Rename methods ``v2`` as ``e`` for classes ``Constraint`` and ``Objective`` - Add benchmark functions -- Improve using of `eval()` in module `omodel` -- Refactor module `interop.andes` as module `interface` for simplicity +- Improve the usage of ``eval()`` in module ``omodel`` +- Refactor module ``interop.andes`` as module ``interface`` for simplicity v0.9.10 (2024-09-03) -------------------- Hotfix of import issue in ``v0.9.9``. -- In module `MatProcessor`, add two parameters `permc_spec` and `use_umfpack` in function `build_ptdf` +- In module ``MatProcessor``, add two parameters ``permc_spec`` and ``use_umfpack`` in function ``build_ptdf`` - Follow RTD's deprecation of Sphinx context injection at build time - In MATPOWER conversion, set devices name as None - Skip macOS tests in azure-pipelines due to failure in fixing its configuration - Prepare to support NumPy v2.0.0, but solvers have unexpected behavior -- Improve the logic of setting `Optz` value +- Improve the logic of setting ``Optz`` value - Support NumPy v2.0.0 v0.9.9 (2024-09-02) @@ -65,11 +70,12 @@ v0.9.9 (2024-09-02) v0.9.8 (2024-06-18) ------------------- -- Assign `MParam.owner` when declaring -- In `MatProcessor`, improve `build_ptdf` and `build_lodf` to allow partial building and incremental building -- Add in 'cases/matpower/Benchmark.json' for benchmark with MATPOWER +- Assign ``MParam.owner`` when declaring +- In ``MatProcessor``, improve ``build_ptdf` and ``build_lodf` to allow partial building and + incremental building +- Add file ``cases/matpower/Benchmark.json`` for benchmark with MATPOWER - Improve known good results test -- Minor fix in `main.py` selftest part +- Minor fix in ``main.py` selftest part - Set dependency NumPy version to be <2.0.0 to avoid CVXPY compatibility issues v0.9.7 (2024-05-24) @@ -84,13 +90,14 @@ References: Frequency Regulation," in IEEE Transactions on Smart Grid, doi: 10.1109/TSG.2024.3356948. - Fix OTDF calculation -- Add parameter `dtype='float64'` and `no_store=False` in `MatProcessor` PTDF, LODF, and OTDF calculation, to save memory -- Add placeholder parameter `Bus.type` +- Add parameter ``dtype='float64'`` and ``no_store=False`` in ``MatProcessor`` PTDF, LODF, and OTDF + calculation, to save memory +- Add placeholder parameter ``Bus.type`` v0.9.6 (2024-04-21) ------------------- -This patch release refactor and improve `MatProcessor`, where it support PTDF, LODF, +This patch release refactor and improve ``MatProcessor`, where it support PTDF, LODF, and OTDF for static analysis. The reference can be found online "PowerWorld > Web Help > Sensitivities > Line @@ -103,11 +110,11 @@ Outage Distribution Factors". - Fix and rerun ex2 - Format ``Routine.get()`` return type to be consistent with input idx type - Remove unused ``Routine.prepare()`` -- Refactor `MatProcessor` to separate matrix building -- Add Var `plf` in `DCPF`, `PFlow`, and `ACOPF` to store the line flow -- Add `build_ptdf`, `build_lodf`, and `build_otdf` +- Refactor ``MatProcessor` to separate matrix building +- Add Var ``plf`` in ``DCPF`, ``PFlow`, and ``ACOPF`` to store the line flow +- Add ``build_ptdf``, ``build_lodf`, and ``build_otdf` - Fix ``Routine.get()`` to support pd.Series type idx input -- Reserve `exec_time` after ``dc2ac()`` +- Reserve ``exec_time`` after ``dc2ac()`` - Adjust kloss to fix ex2 v0.9.5 (2024-03-25) @@ -115,10 +122,10 @@ v0.9.5 (2024-03-25) - Add more plots in demo_AGC - Improve line rating adjustment -- Adjust static import sequence in `models.__init__.py` +- Adjust static import sequence in ``models.__init__.py`` - Adjust pjm5bus case line rate_a - Fix formulation of constraint line angle diff -- Align slack bus angle to zero in `DCOPF` +- Align slack bus angle to zero in ``DCOPF`` - Align StaticGen idx sequence with converted MATPOWER case - Fix several issues in MATPOWER converter From 5efcf8cea87480e0a2fcf73083305701e7ba35e7 Mon Sep 17 00:00:00 2001 From: jinningwang Date: Sat, 23 Nov 2024 12:37:26 -0500 Subject: [PATCH 86/96] Typo --- docs/source/release-notes.rst | 50 +++++++++++++++++------------------ 1 file changed, 25 insertions(+), 25 deletions(-) diff --git a/docs/source/release-notes.rst b/docs/source/release-notes.rst index f0c4f0f0..8f19c569 100644 --- a/docs/source/release-notes.rst +++ b/docs/source/release-notes.rst @@ -14,8 +14,8 @@ v0.9.12 (202x-xx-xx) - Refactor ``OModel.initialized`` as a property method - Add a demo to show using ``Constraint.e`` for debugging -- Fix ``ams.opt.omodel.Param.evaluate()`` when its value is a number -- Improve ``ams.opt.omodel.ExpressionCalc()`` for better performance +- Fix ``ams.opt.omodel.Param.evaluate`` when its value is a number +- Improve ``ams.opt.omodel.ExpressionCalc`` for better performance - Refactor module ``ams.opt`` - Add class ``ams.opt.Expression`` - Switch from PYPOWER to ANDES in routine ``PFlow`` @@ -38,15 +38,15 @@ v0.9.11 (2024-11-14) - Add parameter ``uf`` for enforced generator on/off status - In servicee ``LoadScale``, consider load online status - Consider line online status in routine ``ED`` -- Add methods ``evaluate`` and ``finalize`` in the class ``OModel`` to handle optimization - elements generation and assembling -- Refactor ``OModel.init()`` and ``Routine.init()`` -- Add ANDES paper as a citation file for now +- Add methods ``evaluate`` and ``finalize`` in the class ``OModel`` to handle optimization + elements generation and assembling +- Refactor ``OModel.init`` and ``Routine.init`` +- Add ANDES paper as the citation file for now - Add more routine tests for generator trip, line trip, and load trip - Add a README to overview built-in cases - Rename methods ``v2`` as ``e`` for classes ``Constraint`` and ``Objective`` - Add benchmark functions -- Improve the usage of ``eval()`` in module ``omodel`` +- Improve the usage of ``eval`` in module ``omodel`` - Refactor module ``interop.andes`` as module ``interface`` for simplicity v0.9.10 (2024-09-03) @@ -65,17 +65,17 @@ Hotfix of import issue in ``v0.9.9``. v0.9.9 (2024-09-02) ------------------- -**IMPORTANT NOTICE: This version has known issues and should be avoided.** +**NOTICE: This version has known issues and has been yanked on PyPI.** v0.9.8 (2024-06-18) ------------------- - Assign ``MParam.owner`` when declaring -- In ``MatProcessor``, improve ``build_ptdf` and ``build_lodf` to allow partial building and - incremental building +- In ``MatProcessor``, improve ``build_ptdf`` and ``build_lodf`` to allow partial building and + incremental building - Add file ``cases/matpower/Benchmark.json`` for benchmark with MATPOWER - Improve known good results test -- Minor fix in ``main.py` selftest part +- Minor fix in ``main.py`` selftest part - Set dependency NumPy version to be <2.0.0 to avoid CVXPY compatibility issues v0.9.7 (2024-05-24) @@ -91,30 +91,30 @@ Frequency Regulation," in IEEE Transactions on Smart Grid, doi: 10.1109/TSG.2024 - Fix OTDF calculation - Add parameter ``dtype='float64'`` and ``no_store=False`` in ``MatProcessor`` PTDF, LODF, and OTDF - calculation, to save memory + calculation, to save memory - Add placeholder parameter ``Bus.type`` v0.9.6 (2024-04-21) ------------------- -This patch release refactor and improve ``MatProcessor`, where it support PTDF, LODF, +This patch release refactor and improve ``MatProcessor``, where it support PTDF, LODF, and OTDF for static analysis. The reference can be found online "PowerWorld > Web Help > Sensitivities > Line Outage Distribution Factors". - Refactor DCPF, PFlow, and ACOPF -- Add a loss factor in ``RTED.dc2ac()`` -- Add ``DCOPF.dc2ac()`` +- Add a loss factor in ``RTED.dc2ac`` +- Add ``DCOPF.dc2ac`` - Fix OModel parse status to ensure no_parsed params can be updated - Fix and rerun ex2 -- Format ``Routine.get()`` return type to be consistent with input idx type -- Remove unused ``Routine.prepare()`` -- Refactor ``MatProcessor` to separate matrix building -- Add Var ``plf`` in ``DCPF`, ``PFlow`, and ``ACOPF`` to store the line flow -- Add ``build_ptdf``, ``build_lodf`, and ``build_otdf` -- Fix ``Routine.get()`` to support pd.Series type idx input -- Reserve ``exec_time`` after ``dc2ac()`` +- Format ``Routine.get`` return type to be consistent with input idx type +- Remove unused ``Routine.prepare`` +- Refactor ``MatProcessor`` to separate matrix building +- Add Var ``plf`` in ``DCPF``, ``PFlow``, and ``ACOPF`` to store the line flow +- Add ``build_ptdf``, ``build_lodf``, and ``build_otdf`` +- Fix ``Routine.get`` to support pd.Series type idx input +- Reserve ``exec_time`` after ``dc2ac`` - Adjust kloss to fix ex2 v0.9.5 (2024-03-25) @@ -140,7 +140,7 @@ v0.9.3 (2024-03-06) ------------------- - Major improvemets on demo_AGC -- Bug fix in ``RTED.dc2ac()`` +- Bug fix in ``RTED.dc2ac`` v0.9.2 (2024-03-04) ------------------- @@ -166,7 +166,7 @@ v0.9.0 (2024-02-27) - Fix ``addService``, ``addVars`` - Rename ``RoutineModel`` to ``RoutineBase`` for better naming - Fix ANDES file converter issue -- Initial release to conda-forge +- Initial release on conda-forge v0.8.5 (2024-01-31) ------------------- @@ -182,7 +182,7 @@ v0.8.4 (2024-01-30) v0.8.3 (2024-01-30) ------------------- -- Initial release to PyPI +- Initial release on PyPI v0.8.2 (2024-01-30) ------------------- From 6b210bf222bc45d9b3fda62a7f3c63e5a318bf14 Mon Sep 17 00:00:00 2001 From: jinningwang Date: Sat, 23 Nov 2024 12:40:12 -0500 Subject: [PATCH 87/96] Specify multiprocess<=0.70.16 in requirements as 0.70.17 run into error on Linux --- docs/source/release-notes.rst | 1 + requirements.txt | 1 + 2 files changed, 2 insertions(+) diff --git a/docs/source/release-notes.rst b/docs/source/release-notes.rst index 8f19c569..4f52de57 100644 --- a/docs/source/release-notes.rst +++ b/docs/source/release-notes.rst @@ -22,6 +22,7 @@ v0.9.12 (202x-xx-xx) - Switch from PYPOWER to regular formulation in routine ``DCPF`` - Refactor routines ``DCPF`` and ``DCOPF`` - In ``RDocumenter``, set Srouce to be owner if there is no src +- Specify ``multiprocess<=0.70.16`` in requirements as 0.70.17 does not support Linux RC1 ~~~~ diff --git a/requirements.txt b/requirements.txt index 5ceafe9f..6c6fe056 100644 --- a/requirements.txt +++ b/requirements.txt @@ -10,3 +10,4 @@ openpyxl andes>=1.8.7 pybind11 cvxpy +multiprocess<=0.70.16 From e46c2e09ab839eaa7cd08b01955446c4378cd4ff Mon Sep 17 00:00:00 2001 From: jinningwang Date: Sat, 23 Nov 2024 13:08:48 -0500 Subject: [PATCH 88/96] Format --- ams/routines/dcpf.py | 4 +--- 1 file changed, 1 insertion(+), 3 deletions(-) diff --git a/ams/routines/dcpf.py b/ams/routines/dcpf.py index e99a59ca..8462bb9a 100644 --- a/ams/routines/dcpf.py +++ b/ams/routines/dcpf.py @@ -137,14 +137,12 @@ def unpack(self, **kwargs): idx = var.owner.get_idx() except AttributeError: idx = var.owner.idx.v - else: - pass + # NOTE: only unpack the variables that are in the model or group try: var.owner.set(src=var.src, idx=idx, attr='v', value=var.v) except (KeyError, TypeError): logger.error(f'Failed to unpack <{var}> to <{var.owner.class_name}>.') - pass # --- solver ExpressionCalc results to routine algeb --- for _, exprc in self.exprcs.items(): From e57dff1aa35a0f1e52eef8f39a07cd748685a36f Mon Sep 17 00:00:00 2001 From: jinningwang Date: Sat, 23 Nov 2024 13:21:28 -0500 Subject: [PATCH 89/96] Format --- ams/core/service.py | 62 ++++++++++++++++++++++++++++++--------------- 1 file changed, 41 insertions(+), 21 deletions(-) diff --git a/ams/core/service.py b/ams/core/service.py index 52015f9c..2520cce0 100644 --- a/ams/core/service.py +++ b/ams/core/service.py @@ -33,8 +33,6 @@ class RBaseService(BaseService, Param): Description. vtype : Type, optional Variable type. - model : str, optional - Model name. no_parse: bool, optional True to skip parsing the service. sparse: bool, optional @@ -106,8 +104,8 @@ class ValueService(RBaseService): Description. vtype : Type, optional Variable type. - model : str, optional - Model name. + no_parse: bool, optional + True to skip parsing the service. sparse: bool, optional True to return output as scipy csr_matrix. """ @@ -155,8 +153,8 @@ class ROperationService(RBaseService): Description. vtype : Type, optional Variable type. - model : str, optional - Model name. + no_parse: bool, optional + True to skip parsing the service. sparse: bool, optional True to return output as scipy csr_matrix. """ @@ -194,6 +192,8 @@ class LoadScale(ROperationService): Unit. info : str, optional Description. + no_parse: bool, optional + True to skip parsing the service. sparse: bool, optional True to return output as scipy csr_matrix. """ @@ -236,7 +236,7 @@ class NumOp(ROperationService): Note that the scalar output is converted to a 1D array. - The optional kwargs are passed to the input function. + The `rargs` are passed to the input function. Parameters ---------- @@ -397,8 +397,6 @@ class NumOpDual(NumOp): Description. vtype : Type, optional Variable type. - model : str, optional - Model name. rfun : Callable, optional Function to apply to the output of ``fun``. rargs : dict, optional @@ -407,6 +405,8 @@ class NumOpDual(NumOp): Expand the dimensions of the output array along a specified axis. array_out : bool, optional Whether to force the output to be an array. + no_parse: bool, optional + True to skip parsing the service. sparse: bool, optional True to return output as scipy csr_matrix. """ @@ -467,6 +467,10 @@ class MinDur(NumOpDual): Unit. info : str, optional Description. + vtype : Type, optional + Variable type. + no_parse: bool, optional + True to skip parsing the service. sparse: bool, optional True to return output as scipy csr_matrix. """ @@ -537,8 +541,12 @@ class NumHstack(NumOp): Description. vtype : Type, optional Variable type. - model : str, optional - Model name. + rfun : Callable, optional + Function to apply to the output of ``fun``. + rargs : dict, optional + Keyword arguments to pass to ``rfun``. + no_parse: bool, optional + True to skip parsing the service. sparse: bool, optional True to return output as scipy csr_matrix. """ @@ -624,8 +632,12 @@ class ZonalSum(NumOp): Description. vtype : Type Variable type. - model : str - Model name. + rfun : Callable, optional + Function to apply to the output of ``fun``. + rargs : dict, optional + Keyword arguments to pass to ``rfun``. + no_parse: bool, optional + True to skip parsing the service. sparse: bool, optional True to return output as scipy csr_matrix. """ @@ -709,6 +721,8 @@ class RTED: Keyword arguments to pass to ``rfun``. array_out : bool, optional Whether to force the output to be an array. + no_parse: bool, optional + True to skip parsing the service. sparse: bool, optional True to return output as scipy csr_matrix. """ @@ -801,8 +815,12 @@ class VarReduction(NumOp): A description of the operation. vtype : Type, optional The variable type. - model : str, optional - The model name associated with the operation. + rfun : Callable, optional + Function to apply to the output of ``fun``. + rargs : dict, optional + Keyword arguments to pass to ``rfun``. + no_parse: bool, optional + True to skip parsing the service. sparse: bool, optional True to return output as scipy csr_matrix. """ @@ -818,13 +836,11 @@ def __init__(self, rfun: Callable = None, rargs: dict = None, no_parse: bool = False, - sparse: bool = False, - **kwargs): + sparse: bool = False,): super().__init__(name=name, tex_name=tex_name, unit=unit, info=info, vtype=vtype, u=u, fun=None, rfun=rfun, rargs=rargs, - no_parse=no_parse, sparse=sparse, - **kwargs) + no_parse=no_parse, sparse=sparse,) self.fun = fun @property @@ -859,8 +875,12 @@ class RampSub(NumOp): Description. vtype : Type Variable type. - model : str - Model name. + rfun : Callable, optional + Function to apply to the output of ``fun``. + rargs : dict, optional + Keyword arguments to pass to ``rfun``. + no_parse: bool, optional + True to skip parsing the service. sparse: bool, optional True to return output as scipy csr_matrix. """ From b72a57162525b2a3d22ea9e339615f6ef7ee184c Mon Sep 17 00:00:00 2001 From: jinningwang Date: Sat, 23 Nov 2024 13:25:07 -0500 Subject: [PATCH 90/96] Format --- ams/routines/pflow.py | 63 ++++++++++++++++++++++++++++++++++++++--- ams/routines/routine.py | 8 ++---- 2 files changed, 61 insertions(+), 10 deletions(-) diff --git a/ams/routines/pflow.py b/ams/routines/pflow.py index 34d6c145..3ae30ab1 100644 --- a/ams/routines/pflow.py +++ b/ams/routines/pflow.py @@ -2,8 +2,11 @@ Power flow routines independent from PYPOWER. """ import logging +from typing import Optional, Union, Type from collections import OrderedDict +import numpy as np + from andes.utils.misc import elapsed from ams.core.param import RParam @@ -185,14 +188,66 @@ def enable(self, name): def disable(self, name): raise NotImplementedError - def addRParam(self, **kwargs): + def addRParam(self, + name: str, + tex_name: Optional[str] = None, + info: Optional[str] = None, + src: Optional[str] = None, + unit: Optional[str] = None, + model: Optional[str] = None, + v: Optional[np.ndarray] = None, + indexer: Optional[str] = None, + imodel: Optional[str] = None,): + """ + Not supported! + """ raise NotImplementedError - def addService(self, **kwargs): + def addService(self, + name: str, + value: np.ndarray, + tex_name: str = None, + unit: str = None, + info: str = None, + vtype: Type = None,): + """ + Not supported! + """ raise NotImplementedError - def addConstrs(self, **kwargs): + def addConstrs(self, + name: str, + e_str: str, + info: Optional[str] = None, + is_eq: Optional[str] = False,): + """ + Not supported! + """ raise NotImplementedError - def addVars(self, **kwargs): + def addVars(self, + name: str, + model: Optional[str] = None, + shape: Optional[Union[int, tuple]] = None, + tex_name: Optional[str] = None, + info: Optional[str] = None, + src: Optional[str] = None, + unit: Optional[str] = None, + horizon: Optional[RParam] = None, + nonneg: Optional[bool] = False, + nonpos: Optional[bool] = False, + cplx: Optional[bool] = False, + imag: Optional[bool] = False, + symmetric: Optional[bool] = False, + diag: Optional[bool] = False, + psd: Optional[bool] = False, + nsd: Optional[bool] = False, + hermitian: Optional[bool] = False, + boolean: Optional[bool] = False, + integer: Optional[bool] = False, + pos: Optional[bool] = False, + neg: Optional[bool] = False,): + """ + Not supported! + """ raise NotImplementedError diff --git a/ams/routines/routine.py b/ams/routines/routine.py index d576007e..1abbef33 100644 --- a/ams/routines/routine.py +++ b/ams/routines/routine.py @@ -814,8 +814,7 @@ def addService(self, tex_name: str = None, unit: str = None, info: str = None, - vtype: Type = None, - model: str = None,): + vtype: Type = None,): """ Add `ValueService` to the routine. @@ -833,8 +832,6 @@ def addService(self, Description. vtype : Type, optional Variable type. - model : str, optional - Model name. """ item = ValueService(name=name, tex_name=tex_name, unit=unit, info=info, @@ -850,8 +847,7 @@ def addConstrs(self, name: str, e_str: str, info: Optional[str] = None, - is_eq: Optional[str] = False, - ): + is_eq: Optional[str] = False,): """ Add `Constraint` to the routine. to the routine. From 12676ec24b39f95221c95fb9c7dbbecbf8768faf Mon Sep 17 00:00:00 2001 From: jinningwang Date: Sat, 23 Nov 2024 13:33:28 -0500 Subject: [PATCH 91/96] Refactor ExpressionCalc and Expression __repr__ to reduce duplication --- ams/opt/exprcalc.py | 3 --- ams/opt/optbase.py | 3 +++ 2 files changed, 3 insertions(+), 3 deletions(-) diff --git a/ams/opt/exprcalc.py b/ams/opt/exprcalc.py index f946f43e..430d8fef 100644 --- a/ams/opt/exprcalc.py +++ b/ams/opt/exprcalc.py @@ -137,6 +137,3 @@ def e(self): except Exception as e: logger.error(f"Error in calculating expr <{self.name}>.\n{e}") return None - - def __repr__(self): - return f'{self.__class__.__name__}: {self.name}' diff --git a/ams/opt/optbase.py b/ams/opt/optbase.py index c9773826..b4135a02 100644 --- a/ams/opt/optbase.py +++ b/ams/opt/optbase.py @@ -150,3 +150,6 @@ def size(self): else: logger.warning(f'Routine <{self.rtn.class_name}> is not initialized yet.') return None + + def __repr__(self): + return f'{self.__class__.__name__}: {self.name}' From 3c28bb2cf928b6565627d0be608783d89549fccd Mon Sep 17 00:00:00 2001 From: jinningwang Date: Sat, 23 Nov 2024 13:33:43 -0500 Subject: [PATCH 92/96] Format --- ams/opt/expression.py | 3 --- 1 file changed, 3 deletions(-) diff --git a/ams/opt/expression.py b/ams/opt/expression.py index a80696c0..d7d43c10 100644 --- a/ams/opt/expression.py +++ b/ams/opt/expression.py @@ -198,6 +198,3 @@ def e(self): except Exception as e: logger.error(f"Error in calculating expr <{self.name}>.\n{e}") return None - - def __repr__(self): - return f'{self.__class__.__name__}: {self.name}' From 50efe1599e69d1e3e7db7eb0dfe945e3c0e50296 Mon Sep 17 00:00:00 2001 From: jinningwang Date: Sat, 23 Nov 2024 13:38:01 -0500 Subject: [PATCH 93/96] Format --- ams/opt/omodel.py | 7 ------- 1 file changed, 7 deletions(-) diff --git a/ams/opt/omodel.py b/ams/opt/omodel.py index 8c254aa6..e0aca8b9 100644 --- a/ams/opt/omodel.py +++ b/ams/opt/omodel.py @@ -149,13 +149,6 @@ class OModel(OModelBase): def __init__(self, routine): OModelBase.__init__(self, routine) - @property - def initialized(self): - """ - Return the initialization status. - """ - return self.parsed and self.evaluated and self.finalized - @ensure_symbols def parse(self, force=False): """ From 96c3ebfcbb8d6773a34a43cdd9110b0d2d81048f Mon Sep 17 00:00:00 2001 From: jinningwang Date: Sat, 23 Nov 2024 13:44:26 -0500 Subject: [PATCH 94/96] Typo --- tests/test_rtn_rted.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/tests/test_rtn_rted.py b/tests/test_rtn_rted.py index f112ada7..8dcd650d 100644 --- a/tests/test_rtn_rted.py +++ b/tests/test_rtn_rted.py @@ -81,7 +81,7 @@ def test_set_load(self): def test_dc2ac(self): """ - Test `RTED.init()` method. + Test `RTED.dc2ac()` method. """ self.ss.RTED.dc2ac() self.assertTrue(self.ss.RTED.converted, "AC conversion failed!") From 5c1b47f36f26ca8121b0c978c574792fb35c8773 Mon Sep 17 00:00:00 2001 From: jinningwang Date: Sat, 23 Nov 2024 13:44:48 -0500 Subject: [PATCH 95/96] Remove implementation of DCOPF.dc2ac --- ams/routines/dcopf.py | 53 ------------------------------------------- 1 file changed, 53 deletions(-) diff --git a/ams/routines/dcopf.py b/ams/routines/dcopf.py index 1fb0ed5f..846ff0dd 100644 --- a/ams/routines/dcopf.py +++ b/ams/routines/dcopf.py @@ -192,56 +192,3 @@ def run(self, **kwargs): A custom solve method to use. """ return super().run(**kwargs) - - def dc2ac(self, kloss=1.0, **kwargs): - """ - Convert the DCOPF results with ACOPF. - - Parameters - ---------- - kloss : float, optional - The loss factor for the conversion. Defaults to 1.2. - """ - exec_time = self.exec_time - if self.exec_time == 0 or self.exit_code != 0: - logger.warning(f'{self.class_name} is not executed successfully, quit conversion.') - return False - - # --- ACOPF --- - # scale up load - pq_idx = self.system.StaticLoad.get_idx() - pd0 = self.system.StaticLoad.get(src='p0', attr='v', idx=pq_idx).copy() - qd0 = self.system.StaticLoad.get(src='q0', attr='v', idx=pq_idx).copy() - self.system.StaticLoad.set(src='p0', idx=pq_idx, attr='v', value=pd0 * kloss) - self.system.StaticLoad.set(src='q0', idx=pq_idx, attr='v', value=qd0 * kloss) - - ACOPF = self.system.ACOPF - # run ACOPF - ACOPF.run() - # self.exec_time += ACOPF.exec_time - # scale load back - self.system.StaticLoad.set(src='p0', idx=pq_idx, value=pd0) - self.system.StaticLoad.set(src='q0', idx=pq_idx, value=qd0) - if not ACOPF.exit_code == 0: - logger.warning(' did not converge, conversion failed.') - # NOTE: mock results to fit interface with ANDES - self.vBus = ACOPF.vBus - self.vBus.optz.value = np.ones(self.system.Bus.n) - self.aBus.optz.value = np.zeros(self.system.Bus.n) - return False - self.pg.optz.value = ACOPF.pg.v - - # NOTE: mock results to fit interface with ANDES - self.addVars(name='vBus', - info='Bus voltage', unit='p.u.', - model='Bus', src='v',) - self.vBus.parse() - self.vBus.optz.value = ACOPF.vBus.v - self.aBus.optz.value = ACOPF.aBus.v - self.exec_time = exec_time - - # --- set status --- - self.system.recent = self - self.converted = True - logger.warning(f'<{self.class_name}> converted to AC.') - return True From d8c7b38c8c227f03317754ed4e7d233e9495b9d5 Mon Sep 17 00:00:00 2001 From: jinningwang Date: Sat, 23 Nov 2024 13:47:09 -0500 Subject: [PATCH 96/96] Minor fix --- ams/opt/optbase.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/ams/opt/optbase.py b/ams/opt/optbase.py index b4135a02..56940242 100644 --- a/ams/opt/optbase.py +++ b/ams/opt/optbase.py @@ -58,7 +58,7 @@ def wrapper(self, *args, **kwargs): self.parse() except Exception as e: logger.error(f"Error during initialization or parsing: {e}") - raise + raise e return func(self, *args, **kwargs) return wrapper