Skip to content
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

Introduce GAMS I/O for IXMP4Backend #558

Draft
wants to merge 4 commits into
base: enh/ixmp4-platform-functions
Choose a base branch
from

Conversation

glatterf42
Copy link
Member

This PR adds the next step of functionality for the new IXMP4Backend: I/O for GAMS. I can't yet confirm that the functions here cover all functionality, but locally, the scenario.solve() step in make_dantzig() works.
The PR also adds delete_item() to IXMP4Backend since that was still needed before getting to the GAMS I/O. The GAMSModel always want to record the package versions and remove the sets later on. As it turns out, we don't need to store the package versions in an ixmp4 DB, but I only noticed that after adding the functionality.

The whole make_dantzig() function still fails, however, because it's also trying to add timeseries data, which I will look into next. Handling timeseries data is more on a platform-level, I would say, that's why I want to do that in a separate PR.
The commits here are not very clean, either, sorry for that. I figure that's okay since no one else is working on these files at the moment.

How to review

TBD

PR checklist

  • Continuous integration checks all ✅
  • Add or expand tests; coverage checks both ✅
  • Add, expand, or update documentation.
  • Update release notes.

@glatterf42 glatterf42 added the enh New features & functionality label Feb 12, 2025
@glatterf42 glatterf42 requested a review from khaeru February 12, 2025 13:13
@glatterf42 glatterf42 self-assigned this Feb 12, 2025


# TODO instead of checking len(indexset.data), make ixmp4 return None if it's empty?
def _add_indexsets_to_container(
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

These ~6 functions all look very similar to one another. It should be possible to collapse them into one that is not too complex. Should I make an attempt?

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Yes, that's often the case with these because the underlying objects are so similar. I didn't try to make this more abstract yet since I don't know which form we want them to take eventually. For now, I mainly tried to adhere to the dictum that each function should just have one specific task.
I don't think there's any urgency with this, but if you feel inspired, please go ahead :)

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Understood—I'm basically thinking ahead to trying to review and merge this code, and later maintain it. Despite our best efforts it will probably get pretty big, so there will be a lot to 'grok'.

When looking at it from a high level, it can be easier to understand "There's this one function that does [thing] (for a few different kinds of inputs)" than "There are six functions, what do they each do?"

Let me try for a few minutes and see where I get.

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

You are much more experienced than me, so I'm happy to go along with your solution, but I'm not sure I agree completely. I think having different functions is costly in two ways: writing more lines of code and maintaining similar functions separately.
However, writing dedicated functions is easier to begin with, I would say, compared to starting with the abstraction, and already done. And maintaining them is only costly if gams changes something that requires identical changes in all functions. And even then, one can easily copy-paste the edits and the tests should catch all places where they need to go. If, on the other hand, gams changes something to only e.g. gt.Set, the abstract function requires additional checks to handle these cases, making it more complex.

Either way, thanks for the commit :)
I think the klass dict is missing Equation and I also tried to improve the type hints, I'll push a fixup commit :)

Comment on lines +362 to +365
if isinstance(item, (IndexSet, Scalar)):
return None
else:
return item.indexsets
Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I find this curious: when using return None if isinstance(item, (IndexSet, Scalar)) else item.indexsets, mypy complains that IndexSet and Scalar don't have the attribute .indexsets, but writing it like this is fine. This almost seems like a mypy limitation, maybe it's even worth an issue with them.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

@khaeru khaeru added the backend.ixmp4 Interaction with ixmp4 via IXMP4Backend label Mar 26, 2025
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
backend.ixmp4 Interaction with ixmp4 via IXMP4Backend enh New features & functionality
Projects
None yet
Development

Successfully merging this pull request may close these issues.

2 participants