-
Notifications
You must be signed in to change notification settings - Fork 6
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
DRY optimization core layer #96
Conversation
@glatterf42 OK so i think the currently proposed changes (removing the commented base model) are good. I also think the rest of the facade layer could also benefit from something like this. Lets figure the |
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.
noticed that i never approved this
2698897
to
d964068
Compare
Codecov ReportAttention: Patch coverage is
Additional details and impacted files@@ Coverage Diff @@
## main #96 +/- ##
=======================================
- Coverage 86.9% 86.9% -0.1%
=======================================
Files 228 230 +2
Lines 8149 8156 +7
=======================================
+ Hits 7089 7095 +6
- Misses 1060 1061 +1
|
Please note: with d964068, this PR is fine to be merged to However, it adds a lot of untyped code, so I will rebase it onto #129's branch and resolve all merge conflicts/add type hints here, too. In fact, I'll even use #138's branch since that makes even more relevant changes. Once that works, we can merge it; should it turn out to be too complicated, we can still revert to the commit mentioned above. |
d964068
to
74e773f
Compare
50d6fae
to
79b2c14
Compare
79b2c14
to
bc61725
Compare
* Spell out signature of create() to give proper type hints
74e773f
to
abab5a5
Compare
* Add abstraction layer for core optimization reusability * Add annotations
For #94, @danielhuppmann asked in the
core
layer: "Why doesParameter
not inherit fromTable
"? And sure enough, these optimization items share a lot of common functionality. So before moving on toVariable
andEquation
, for which this will be the case even more so, I wanted to take another look at this.@meksor, I expect you'll be the one to take a look at this. Sorry, this description might be a bit lengthy.
Initially, I interpreted the question as for the DB layer, and how to handle inheritance (or rather Composition) there has been recently discussed in #82. Instead, this PR focuses on the core layer, where we also have two classes per optimization item that could potentially benefit from higher reusability (to follow the DRY principle): e.g., the
Table
(mostly a collection of data attributes) and theTableRespository
(functions to handleTable
s).For the
Table
s themselves (and similar), we are currently using Python's property to make data attributes accessible in a controlled way. Unfortunately, theseproperty
s don't mix well with inheritance. This article describes how to still make it work for thegetter
part, but since we would still have to repeat the same number of lines in the data files, I'm not sure it's worth the effort here. What we stand to gain is a separation of functionality and placeholder code, it seems to me. So when we want to make changes to how e.g.data
is returned from an optimization model, we would just need to change one place and all others would inherit these changes. However, code repetition would not be decreased and understanding the models might become more complicated, so I've put this on hold for now.For
TableRepository
, there is luckily a quite straightforward way: we can employ the same approach as in the DB layer and create anOptimizationBaseRepository
in the core layer that can be extended withCreator
,Retriever
, etc as needed. This seems to work well and every optimization item repo then just needs to clarify which_model_type
and_backend_repository
it is interested in.For better understanding, I wanted to type hint the
_backend_repository
in theOptimizationBaseRepository
. However, we didn't already have an appropriate object to do so. That's why I have created aBackendBaseRepository
inabstract.optimization.base
, which simply announces how the actualTableRepository
in the DB layer is going to look. I'm not sure if this is the correct location; I've usually worked with abstract models to type hint API/DB layer functionality, but I also don't know where else it would live.This is the point where I'm not sure I'm naming things correctly anymore. I started from
Table
, which already hasTable.data
, which is arguably the largest part of shared functionality, at least in the DB layer, and which is missing fromIndexSet
andScalar
. So these two items could not use theOptimizationBaseRepository
as it is now, which seems like a misnomer. Maybe the data functionality needs to be further separated (allowing us to DRY even more parts of this layer) or maybe we find another name?The same applies for
BackendBaseRepository
.Note
For the curious: there's an issue on CPython about
property
s and inheritance from 2012.