-
Notifications
You must be signed in to change notification settings - Fork 238
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Add support for Greybox model DOF counitng in degrees_of_freedom function #1512
Conversation
Codecov ReportAttention: Patch coverage is
Additional details and impacted files@@ Coverage Diff @@
## main #1512 +/- ##
=======================================
Coverage 77.06% 77.07%
=======================================
Files 387 387
Lines 62486 62576 +90
Branches 10236 10259 +23
=======================================
+ Hits 48156 48231 +75
- Misses 11896 11908 +12
- Partials 2434 2437 +3 ☔ View full report in Codecov by Sentry. |
idaes/core/util/model_statistics.py
Outdated
) + number_grey_box_equalities(block) | ||
|
||
|
||
def number_grey_box_equalities(block): |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
number_of_X
is less ambiguous in English.
Can add -> int
to indicate return type and, if it's possible, the expected type of block
.
In docs: "Method to" is superfluous and at any rate this is a function.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
done
idaes/core/util/model_statistics.py
Outdated
|
||
def number_grey_box_equalities(block): | ||
""" | ||
Method to return the number of equality constraints in GreyBox |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I would say, "Compute total number of equality constraints for all GreyBox objects in this block."
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
done
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Thanks for this, it is a case we have not considered and Greybox models are becoming more important. To start with, I had a few suggestions for more tests and modularization of the code.
However, a more significant request would be to add these tools and (and some test cases) into the diagnostics toolbox as well. We should at a minimum check that the DoF and model statistics output (number of variables and constraints) accurately reflect the model with a Greybox present. It might also be nice to add additional sub-rows in the model statistics for greybox components when they are present. Finally, we should look for any edge cases that might occur with the rest of the toolbox and if there are other things we need to consider when using greybox models.
idaes/core/util/model_statistics.py
Outdated
@@ -529,7 +555,7 @@ def deactivated_inequalities_generator(block): | |||
block : model to be studied | |||
|
|||
Returns: | |||
A generator which returns all indeactivated equality Constraint | |||
A generator which returns all in deactivated equality Constraint |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
The "in" should be removed here.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
done
idaes/core/util/model_statistics.py
Outdated
# Checks for greyboxes, and if they exist will add | ||
# input and output vars to var_set if they are free | ||
# inputs and outputs are defined names for greybox class and should always exist | ||
for grey_box in _iter_indexed_block_data_objects( |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I think it would be good to have a number_of_greybox_variables
function to do this so that users can check that separately and so that we can test the code in isolation.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
added one for variables, and unfixed variables.
@@ -685,6 +689,42 @@ def test_degrees_of_freedom(m): | |||
assert degrees_of_freedom(m.b2) == -1 | |||
|
|||
|
|||
@pytest.mark.unit |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
We should also have test for the functions to collect the number of greybox variables and constraints.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
added tests
IIRC, outputs just get translated to equality constraints of the form |
@Robbybp : Right, so you can include equality constraints in a graybox and send them to the solver (ipopt) - but the variables in those constraints are not actually in pyomo space- and thus we can't count them (hence why here I am only looking at inputs and outputs and ignoring any internal equality constraints). (e.g. looking at this example) At this point, we just want to look at the graybox as a constraint that binds exposed pyomo inputs to outputs on it, and assume that the users properly setup the internal graybox model to have the correct DOFS... ideally we would have some metadata on the graybox that tells us its internal # of DOFs so we can include it in the total count. But I opt to avoid that for now... unless we want to add support for this now. |
idaes/core/util/model_statistics.py
Outdated
equalities = 0 | ||
for grey_box in _iter_indexed_block_data_objects( | ||
block, ctype=ExternalGreyBoxBlock, active=True, descend_into=True | ||
): | ||
equalities += len(grey_box.outputs) | ||
return equalities |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
As written, this will be incorrect for grey box models that have a nonzero number of "explicitly defined equality constraints" (as opposed to the equality constraints that are created to define the outputs). I think all that's needed to correct this is:
equalities += grey_box.get_external_model().n_equality_constraints()
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Right, did not think about this scenario! Updated to test graybox with suggestion, this now includes an example constraint that binds 2 inputs together.
def input_names(self):
return ["a1", "a2", "a3"]
def output_names(self):
return ["o1", "o2"]
def equality_constraint_names(self):
return ["a12"]
def evaluate_equality_constraints(self):
a1 = self._input_values[0]
a2 = self._input_values[1]
return [a1 * 0.5 - a2]
So I started adding display to diagnostics tools
|
I'm curious what happens in the structural analysis when you have a greybox model. I can't imagine the under/overconstrained subsystems are handled correctly. Do you know what happens in this case? |
Freeing up my review seeing as I am leaving the project (for now at least).
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Looks mostly good, just a few issues with docstrings to resolve.
@@ -195,7 +294,11 @@ def number_total_constraints(block): | |||
Returns: | |||
Number of Constraint components in block | |||
""" | |||
return sum(1 for _ in activated_block_component_generator(block, ctype=Constraint)) | |||
number_standard_constraints = sum( |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
The docstring of this function needs to be updated to explain what's going on with greybox models.
@@ -280,7 +383,9 @@ def number_deactivated_constraints(block): | |||
Returns: | |||
Number of deactivated Constraint components in block | |||
""" | |||
return sum(1 for _ in deactivated_constraints_generator(block)) | |||
standard_equalities = sum(1 for _ in deactivated_constraints_generator(block)) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
The docstring of this function needs to be updated to explain what's going on with greybox models.
@@ -325,7 +430,9 @@ def number_total_equalities(block): | |||
Returns: | |||
Number of equality Constraint components in block | |||
""" | |||
return sum(1 for _ in total_equalities_generator(block)) | |||
standard_equalities = sum(1 for _ in total_equalities_generator(block)) | |||
greybox_equalities = number_activated_greybox_equalities(block) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
The docstring of this function needs to be updated to explain what's going on with greybox models.
@@ -377,7 +484,53 @@ def number_activated_equalities(block): | |||
Returns: | |||
Number of activated equality Constraint components in block | |||
""" | |||
return sum(1 for _ in activated_equalities_generator(block)) | |||
return sum( |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
The docstring of this function needs to be updated to explain what's going on with greybox models.
|
||
def number_activated_greybox_equalities(block) -> int: | ||
""" | ||
Function to compute total number of equality constraints for all GreyBox objects in this block. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
"total number of equality constraints" -> "total number of implied equality constraints"
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Actually, it might be fine leaving as is. I didn't realize GreyBox models have an n_equality_constraints()
attribute. The distinction here is probably that they don't exist as Constraint
objects.
idaes/core/util/model_statistics.py
Outdated
A GreyBox model is always assumed to be 0DOFs where each output[i]==f(inputs) | ||
where f is GreyBox model, this should be true regardless if | ||
GreyBox model is doing internal optimization or not, as every output | ||
is calculated through a the GreyBox internal model using provided inputs. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
"through a the GreyBox" -> "through the GreyBox"
Also, is this 0DOFs property of GreyBox models always true, or is it true only if the user has correctly configured their GreyBox model?
@@ -423,7 +576,9 @@ def number_deactivated_equalities(block): | |||
Returns: | |||
Number of deactivated equality Constraint components in block | |||
""" | |||
return sum(1 for _ in deactivated_equalities_generator(block)) | |||
standard_equalities = sum(1 for _ in deactivated_equalities_generator(block)) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Update docstring to reflect inclusion of Greybox.
idaes/core/util/model_statistics.py
Outdated
A ComponentSet including all unfixed Var components which appear within | ||
activated equality Constraints in block |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This docstring is incorrect. Also, are greybox_variables
Var
objects?
assert number_total_equalities(m) == 5 | ||
assert number_deactivated_equalities(m) == 3 | ||
assert number_deactivated_constraints(m) == 3 | ||
report_statistics(m) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
What does this line test?
You also should add a warning to the structural analysis tool that it's not compatible with grey box models, then make sure that |
oof, I have not worked with this tool set deeply enough to even know where to start. Can we leave this for another PR. At this stage, I doubt many users would try to use diagnostics with Greybox models in general... so it should be okay. I would rather make a new PR after testing and finding all issues with diagnostic tools and then add appropriate doc/warnings. |
I agree, but I don't want to confidently give wrong analyses if we're supporting GreyBox in one part of the tool (model statistics). Can you raise a "beta warning" or something in |
@avdudchenko Can you also open an issue about handling GBBs in structural and numerical diagnostics? If you provide a MWE to test on, I can try to update incidence analysis to do something reasonable. |
Will do. |
I added error raising for those functions for now. |
The We can sort out which methods support grey box models and which ones don't in another issue/PR and fine tune our warnings/errors later. |
…enko/idaes-pse into add_gray_box_do_counting
Done, I also did not remove prior tests as they directly call on report function. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
LGTM
class BasicGrayBox(ExternalGreyBoxModel): | ||
def input_names(self): | ||
return ["a1", "a2", "a3"] | ||
|
||
def output_names(self): | ||
return ["o1", "o2"] | ||
|
||
def equality_constraint_names(self): | ||
return ["a_sum"] | ||
|
||
def evaluate_equality_constraints(self): | ||
a1 = self._input_values[0] | ||
a2 = self._input_values[1] | ||
return [a1 * 0.5 + a2] |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Just wondering if this needs to be replicated within tests like you do now or if you could just create once outside of tests.
Co-authored-by: Adam Atia <[email protected]>
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
LGTM - I had one minor comment that you can consider or deflect. One other suggested change to commit--minor, but affects documentation. Approving regardless so as not to hold things up.
Fixes
The current degrees_of_freedom function does not properly account for GreyBoxModel as a set of constraints, resulting in an incorrect number of DOFs being presented.
Summary/Motivation:
With increased usage of GreyBoxModel (specifically Reaktoro-pse) there is a need to support DOF counting when building IDAES model with GreyBoxModels
Changes proposed in this PR:
-add the capability to add unfixed input and output vars on GreyBox model to ComponentSet produced by unfixed_variables_in_activated_equalities_set
-add a counter for the number of "equalities" in gray box model in number_activated_equalities by adding number_grey_box_equalities method
Counting the number of equalities in GreyBox model:
GrayBox models do not provide traditional Pyomo Constraints that bind inputs to outputs as such, we need to assume how many GrayBox model DOFs there are.
In most cases (at least I can't think of any exceptions) GreyBoxes must be 0DOF in the eyes of Pyomo/IPOPT as each output must be some sort of function of provided inputs, regardless if the internal model in the GreyBox is 0DOF, an optimization, or it exposes internal constraints to solve the problem.
As such, it should be safe to assume that the number of outputs equals the number of equality constraints.
Legal Acknowledgement
By contributing to this software project, I agree to the following terms and conditions for my contribution: