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

Fix recovery date off by one day bug #230

Merged
merged 1 commit into from
Jan 16, 2025
Merged

Conversation

samholmes
Copy link
Contributor

@samholmes samholmes commented Jan 16, 2025

The date picker would return the date picked at the current local time
based on the device's locale (timezone). Because toISOString was used,
the resulting date my be different then the picked date because the day
in UTC may be a different day then what the current device is in.

To solve this, use getDate/Month/FullYear methods which respect locale
to format the ISO formatted date string. This has been abstracted into
a utility.

CHANGELOG

Does this branch warrant an entry to the CHANGELOG?

  • Yes
  • No

Dependencies

none

Description

none

src/util/date.ts Outdated
Comment on lines 1 to 11
/**
* Returns the date portion of a ISO formatted date string (YYYY-MM-DDT...), but
* makes sure to return the date local to the current timezone.
*
* It might be the 15th in the local timezone but the 16th in UTC. So if one
* were to use `toISOString()` then the date would be 1 day off from the local
* timezone.
*
* @param date A JS Date Object
* @returns ISO formatted date string (YYYY-MM-DD) but in the local timezone.
*/
Copy link
Contributor

Choose a reason for hiding this comment

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

I don't think this utility function should exist. Nobody else should be using this, and keeping it around is super-dangerous.

Basically, both JavaScript Date objects and the ISO format operate in seconds-since-the-epoch. This metric is simple, unambiguous, and free of human timezone concerns. When we talk about time in computer systems, this is the way to go, because it's precise.

The recovery questions, on the other hand, do not want a specific point in time. They want string, which we can later check for equality using ===. The string is date-ish, sure, but it's NOT a an "ISO formatted date sting" by any definition of the term. You can't take the recovery answer and say for sure when they got married or whatever the question is - to answer that, you would need to know the timezone, which we don't have. This is because humans, unlike computers, are really sloppy with the way we think about dates. Everything depends on context and culture for us.

As soon as you mix human and computer ways of thinking, you start to get into trouble. You end up with situations where all the values in the database are off by 8 hours, because the server happened to be in GMT+8 or something. The solution is to always do everything in computer time, since that keeps the data free from corrupting influences.

If you need to deal with local time, that's an input/output issue, and should really be using Date.toLocaleString in 99% of the cases. If we are going to to talk to a human, let's talk to the human in their preferred format!

This brings us to the recovery questions, which are a weird case. We want the value to use "human dates", but we don't want the value to use the culture's preferred format - we want YYYY-MM-DD, so === is more likely to work. Note that this is NOT the ISO format, since it's in local time, and yet doesn't include timezone specifier on the end like GMT+8. It's a weird hybrid franken-format used only by the recovery system.

So, we cannot leave this utility sitting around. It implements a dangerous hybrid format that looks like something it's not, and is used only by the recovery system. We need to give this a better name, like toRecoveryDateString, and tuck it away in the recovery system were nobody else is going to find it.

Copy link
Contributor Author

Choose a reason for hiding this comment

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

I agree with all this said. I updated the PR to reflect your comment.

@samholmes samholmes force-pushed the sam/recovery-dates branch 2 times, most recently from be69216 to 80b2282 Compare January 16, 2025 23:27
@@ -116,3 +116,17 @@ export function DateModalAndroid(props: Props) {

export const DateModal =
Platform.OS === 'android' ? DateModalAndroid : withTheme(DateModalIos)

/**
* Returns teh date portion of a date object, formatted YYYY-MM-DD.
Copy link
Contributor

Choose a reason for hiding this comment

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

s/teh/the/

* Returns teh date portion of a date object, formatted YYYY-MM-DD.
* It makes sure to return the date local to the current timezone.
*
* It's only use-case is to convert a date object from the date picker model
Copy link
Contributor

Choose a reason for hiding this comment

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

s/model/modal/

The date picker would return the date picked at the current local time
based on the device's locale (timezone). Because `toISOString` was used,
the resulting date my be different then the picked date because the day
in UTC may be a different day then what the current device is in.

To solve this, use `getDate/Month/FullYear` methods which respect locale
to format the ISO formatted date string. This has been abstracted into
a utility that is only scoped to the DateModel component.
@samholmes samholmes merged commit 81f0c63 into master Jan 16, 2025
1 check passed
@samholmes samholmes deleted the sam/recovery-dates branch January 16, 2025 23:29
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.

2 participants