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

0.4.x: deprecate methods that have an _opt() alternative #827

Merged
merged 3 commits into from
Oct 9, 2022

Conversation

djc
Copy link
Member

@djc djc commented Sep 28, 2022

As a compatible step towards a future for 0.5 where we try to not panic internally, this deprecates all methods that have an _opt() alternative in order to require the caller to unwrap() or expect() instead of the library doing that.

This is a small step towards #815 and allows us to test with current users how this direction is perceived.

@djc djc requested a review from esheppa September 28, 2022 12:22
@djc djc force-pushed the 0.4.x-deprecate-no-opt branch 3 times, most recently from f166892 to 9690533 Compare September 28, 2022 14:26
src/offset/local/unix.rs Outdated Show resolved Hide resolved
@djc djc force-pushed the 0.4.x-deprecate-no-opt branch 2 times, most recently from 11facfa to 2e423e2 Compare September 28, 2022 15:29
Copy link
Collaborator

@esheppa esheppa left a comment

Choose a reason for hiding this comment

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

This looks excellent and I think this is the right incremental approach to get this done. Hopefully we will get some good feedback once 0.4.23 is released which can be incorporated into extensions on this in 0.5.

src/naive/date.rs Outdated Show resolved Hide resolved
src/naive/date.rs Outdated Show resolved Hide resolved
src/offset/fixed.rs Show resolved Hide resolved
src/offset/fixed.rs Show resolved Hide resolved
src/offset/local/unix.rs Outdated Show resolved Hide resolved
src/offset/mod.rs Outdated Show resolved Hide resolved
src/naive/time/mod.rs Outdated Show resolved Hide resolved
src/naive/time/mod.rs Outdated Show resolved Hide resolved
src/naive/time/mod.rs Outdated Show resolved Hide resolved
src/naive/time/mod.rs Outdated Show resolved Hide resolved
@djc djc force-pushed the 0.4.x-deprecate-no-opt branch from 2e423e2 to c5ebbcd Compare October 3, 2022 10:54
@djc
Copy link
Member Author

djc commented Oct 3, 2022

Good feedback, thanks for catching the messy example situation -- I stumbled upon it late and then figured I'd ask for feedback before doing another round of cleaning that up. I think I've removed all the examples from the non-opt methods, and made sure that all _opt() methods have at least one example.

Copy link
Collaborator

@esheppa esheppa left a comment

Choose a reason for hiding this comment

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

Looks good! I've left a few comments in NaiveDate days iterator, also had one general thought:

where there are blocks like:

    /// assert_eq!(NaiveDate::from_ymd_opt(2015, 6, 3).unwrap().succ_opt(),
    ///            Some(NaiveDate::from_ymd_opt(2015, 6, 4).unwrap()));

potentially this style could be used instead:

    /// assert_eq!(NaiveDate::from_ymd_opt(2015, 6, 3).unwrap().succ_opt(),
    ///            NaiveDate::from_ymd_opt(2015, 6, 4)));

however, I don't think its worth doing those changes here as anyway, it may change further if the functions are changed to return result instead in some cases

@@ -1885,7 +1733,7 @@ impl Iterator for NaiveDateDaysIterator {
// current < NaiveDate::MAX from here on:
let current = self.value;
// This can't panic because current is < NaiveDate::MAX:
self.value = current.succ();
self.value = current.succ_opt().unwrap();
Copy link
Collaborator

Choose a reason for hiding this comment

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

could we use self.value = current.succ_opt()?; and then remove the whole if block?

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 played with something like that at the time, but I don't think so? In the case succ_opt() returns None but current is Some we still have to return that last value. Arguably the iterator contract should be changed but I didn't want to take that on as part of this PR.

Copy link
Collaborator

Choose a reason for hiding this comment

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

sounds good - I just remembered why this stood out to me, I've made some changes to it in c7e1b9b anyway, so lets merge this as-is

src/naive/date.rs Show resolved Hide resolved
@djc djc merged commit 7cd5e03 into 0.4.x Oct 9, 2022
@djc djc deleted the 0.4.x-deprecate-no-opt branch October 9, 2022 12:15
@djc djc mentioned this pull request Oct 18, 2022
@hoodie
Copy link

hoodie commented Nov 13, 2022

I have to admit that this makes for some rather verbose code.

@djc
Copy link
Member Author

djc commented Nov 13, 2022

I have to admit that this makes for some rather verbose code.

Can you give some examples?

@hoodie
Copy link

hoodie commented Nov 13, 2022

the examples actually already contain that

let dt: NaiveDateTime = NaiveDate::from_ymd_opt(2016, 7, 8).unwrap().and_hms_opt(9, 10, 11).unwrap();

which used to be

let dt: NaiveDateTime = NaiveDate::from_ymd(2016, 7, 8).and_hms(9, 10, 11);

I completely understand the motivation of having a fallible default that doesn't panic, but for some cases (in my case just example code) it would have been acceptable to have the panicking version as long as it is well documented that it panics.

BTW: my first through when I saw that some of these return LocalResult<T> was "maybe I can just continue because LocalResult has the same functionality with the .and_hms():D "
That would have been cool. At least for a builder pattern

@djc
Copy link
Member Author

djc commented Nov 13, 2022

Maybe that could make sense? Would you be able to submit a PR for something like that?

@esheppa
Copy link
Collaborator

esheppa commented Nov 13, 2022

@hoodie - I'm keen to understand the use cases for constructing dates from literals vs parsing them. If possible could you give a bit more detail about how/why you are using functions like from_ymd_opt and how often that occurs in your code base?. In chrono we construct dates/times like this a lot, mainly in tests and examples, but in my regular code I find myself doing this quite rarely.

In terms of continuing, the Option and Result helper methods mean you can do something like:

let dt: NaiveDateTime = NaiveDate::from_ymd_opt(2016, 7, 8).and_then(|d| d.and_hms_opt(9, 10, 11)).unwrap();

@hoodie
Copy link

hoodie commented Nov 14, 2022

@esheppa you are actually right. I also only use this in examples, which is why I was a bit disappointed that my own example code started looking a bit cumbersome after fixing these deprecations ;)

I don't think this needs to be fixed actually, it just makes example code a bit less sexy now

@tafia
Copy link

tafia commented Nov 15, 2022

Would it be possible to reactivate it under certain conditions?

// not sure it actually works
#[cfg_attr(not(all(test, feature = "opt_unwrap")), deprecate)]

The rationale is

  • this crate is used a LOT and testing a particular date is extremely common. The deprecation add lot of noise either in the code or the tests outputs
  • I believe dates created in tests are overwhelmingly using valid dates and thus the opt is most of the time useless in that scenario
    • we're expecting a valid date
    • if this is not a valid date, test will panic which is actually ok!

EDIT; An alternative would be to use some sentinel value for dates (a bit like float, some not-a-date date). Not sure if this is better it'd just help with most common scenario (also I suppose, like for float may lead to some compiler optimizations?)

@djc
Copy link
Member Author

djc commented Nov 15, 2022

@tafia there is some discussion in #815 about adding macros that could statically verify date/time validity.

I don't think doing a Cargo feature would make sense for this.

@njaard
Copy link

njaard commented Jun 20, 2023

I really don't like the deprecation warning. Now I have to go through all my code and add _opt to hundreds of locations, then when chrono 0.5 comes out, I'll have to do it again, removing the _opt and adding an error check.

Considering that the breaking change in 0.5 will be a compile-time error, could I suggest that the deprecation warning be removed, or made as a feature flag?

@djc
Copy link
Member Author

djc commented Jun 26, 2023

We'll see what we land on for 0.5 -- that will still take some time. For the time being, I don't think we're going to (conditionally) drop the deprecation warning, you can always introduce a wrapper in your own code if you're worried about churn.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

5 participants