-
Notifications
You must be signed in to change notification settings - Fork 342
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 recharge to gdp callback #1223
Conversation
This is on my todo list! Probably will get to it tomorrow @DavidLitwin. |
@@ -752,6 +753,6 @@ def run_with_adaptive_time_step_solver(self, dt): | |||
remaining_time -= substep_dt | |||
self._num_substeps += 1 | |||
|
|||
self._callback_fun(self._grid, substep_dt) | |||
self._callback_fun(self._grid, self.recharge, substep_dt) |
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.
For backward compatibility, I think that this needs to take either 2 or 3 arguments.
-
The docs should state this, and that the two option callback function is deprecated.
-
In the code itself the number of arguments should get tested. One example of testing the number of arguments a function takes is given at line 168 of landlab\components\lithology\litholayers.py @mcflugen may know a better way to do this.
-
If two arguments are provided then a DeprecationWarning should be raised and grid, dt passed. If three, then grid, recharge, dt passed. (Sidenote: Raising the DeprecationWarning will be helpful for whoever manages the next major version because it will help them find that this is a change that was deprecated).
-
Add one test, so you have one test with a three element function and one with a two element function (that yields a deprecation warning). At some point I remember it being tricky to figure out how to test for deprecated warnings. Here is a link to the part of pytest that shows an example.
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 taking a look Katy. I'm trying out the method function.__code__.co_varnames
, and the one problem I see is that it doesn't distinguish between arguments and keyword arguments. There can be as many keyword arguments as the user wants, e.g. to supply the name of the output file. I'm looking into using the signature
class from inspect, but I'm still not quite sure how to separate the two. For example, someone could provide a function something like:
def callback(grid, recharge, dt, d='file.csv'):
return np.save_txt(d, grid['surface_water_specific__discharge']*dt)
And that would be fine.
Will think on this @DavidLitwin... though @mcflugen is the best at this sort of thing. |
@kbarnhart, @mcflugen had some good comments over here, which I've started to implement. Is this the kind of functionality you had in mind? I will try to add a test later today. |
@DavidLitwin it felt easier to make changes/comment by editing the file directly. Feel free to roll back any of my changes if you don't like them. Also, some of the changes are just comments that I expect you to delete. A function may have keyword arguments set in its definition, but the way that the callback function is actually used, there is no mechanism to pass keyword arguments to the function (so defaults are always used, yes). So I think its as easy as testing whether it works with three args, or two args, and then if neither works, raising an actual error. So I think what you have done is 95% there! 👍 LMK if you have any questions about what I've written. |
Hi @kbarnhart, this looks good, thank you! I guess I'm a little unsure on the use of keyword arguments. The way I've been using this thing, I only ever use the defaults of the keyword arguments. But if I add I trust your judgement on the ValueError call, I don't know very much about errors yet. One thing that might be unclear currently is that the variable |
note: I wrote this really fast because I wanted to get you an answer today instead of in two days. so if things don't make sense, assume that is why. re kwds:lets say you define a function with a signature
And you pass that as a function to be used within another function.
Unless the input to `big_function has an input for a dictionary to pass as keyword arguments to the internally called callback function, your function is equivalent to:
Providing the ability to pass keyword arugments might look like:
(you'd do it this way if there might be other functions that you pass keywords too) Or, if the callback function is the only thing you might need to pass keywords to:
Importantly, in both of these examples, there is some source of **kwds... As best as I can tell, in your application there is no mechanism by which kwds are passed into the run_one_step or init such that they can be passed to the callback function. re: value errorMy thinking was, if a callback function couldn't deal with being passed 2 or 3 arguments, then it would break upon use. That is a thing a user should find out as soon as GDP attempts to set the callback function, not upon running. |
@kbarnhart sorry for the delayed reply. Thank you for the lesson on keyword arguments! I've included a way to set them, and added a statement in the final test that makes sure that they are taken. |
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.
@DavidLitwin this looks great. One minor comment which is to make callback_kwds private.
I also have a question fro @mcflugen that I think won't require any changes, but if it does, they'd be minor.
@mcflugen - my question is if there should be anything else done here to verify that no unused keyword arguments are passed. We often use pop, and then check that **kwds is empty at the end of the init... However, in this case, @DavidLitwin saves **callback_kwds and passes them to the callback_fun in its setter (I think this is the right approach). And if there were unused **kwds, then this should raise an error... So I think its OK.
Co-authored-by: Katy Barnhart <[email protected]>
…dlab into dlitwin/gdp_callback
@kbarnhart great, I think I've addressed everything now, save a comment from @mcflugen. |
@DavidLitwin I think this looks good. There was a transient link-check error that caused the docs stage to fail, but other than that all the tests pass! Since I think it would be good for @mcflugen to take a quick look I've added him as a reviewer. |
@mcflugen when you have a chance could you take a look at this? |
@kbarnhart what do you think about going ahead with this? |
done. |
I added recharge rate as argument to the callback function. This is probably the simplest solution to obtaining the recharge rate for each substep when it cannot be guaranteed to exist as a grid field. While recharge rate doesn't change during substeps, I've found it difficult to reconstruct the subtimestep recharge rates when a long series of stochastic recharge events are run.