-
-
Notifications
You must be signed in to change notification settings - Fork 31.5k
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
datetime: min date (0001-01-01 00:00:00) can't be converted to local timestamp #75395
Comments
This worked in Python 3.6.0 and before:
The error output is:
But it used to return Appears to be related to https://bugs.python.org/issue29921 |
This error is a side effect of the implementation of the PEP-495. In your timezone, datetime(1, 1, 1, 0, 0, 0).timestamp() creates an internal timestamp with year=0. The problem is that the internal function ymd_to_ord() doesn't support year=0:
The question is if the result was correct before? |
Note: My change was the commit 5b804b2fb0eaa2bacb4afcfe4cfa85b31475e87f which prevented a crash, bpo-29100. |
That's a valid |
The question is whether -62135658000.0 is the "correct" or even meaningful value: >>> datetime.utcfromtimestamp(-62135658000.0)
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
ValueError: year is out of range (I ran the above in Python 2.7 to avoid any 3.x datetime innovations.) I don't see much of a value in allowing datetime.timestamp() to produce values that correspond to out of bounds datetimes in UTC. In most cases this will only result in error later on in the program, or worse an error passing undetected. What is the use case for datetime.min.timestamp()? I suspect OP encountered this issue in an overly aggressive edge case testing and not in a real user scenario. |
On the second thought, a reasonable design can use datetime.min/max as placeholders for unknown times far in the past/future compensating for the lack datetime ±inf. In this use case, it may be annoying to see errors from timestamp() instead of some ridiculously large values. I am -0 on making this change. If anyone is motivated to produce a patch - I'll review it, but I am not going to write it myself. I am marking this "tests needed" because we need a test complete with setting an appropriate timezone that demonstrates this issue. |
The use case was parsing user input of ISO 8601 date strings and converting them to UNIX epochs. The input "0001-01-01T00:00:00" is valid, parses to a valid |
Ok, so I understand the issue now. |
It your use case, the input "0001-01-01T00:00:00" was in what timezone? I suspect what you want is >>> datetime.min.replace(tzinfo=timezone.utc).timestamp()
-62135596800.0 And not -62135658000.0. If you work with naive datetimes and just want to roundtrip between datetimes and numbers, you should not use a timezone-dependent conversion - attach UTC timezone before calling .timestamp() as shown above. You probably don't want your program to produce different numbers for the same "0001-01-01T00:00:00" input depending on where it is run. |
BTW, I was originally against introducing .timestamp() method and this issue illustrates why it is problematic: people use it without understanding what it does. If you want the distance between datetime.min and datetime(1970,1,1) in seconds - compute it explicitly: >>> (datetime.min - datetime(1970,1,1)) / timedelta(0, 1)
-62135596800.0 If you don't want to loose precision is roundtriping between datetimes and numbers - use microseconds: >>> (datetime.min - datetime(1970,1,1)) // datetime.resolution
-62135596800000000 It is perfectly fine to work with naive datetimes and ignore the timezone issues when all your timestamps are in one timezones, but if you choose to ignore timezones - don't use .timestamp(). |
This is the part where I agree with you. I suspect the error in the UTC case is an artifact of PEP-495 fold calculations that require probing times ±24 hours around the time being converted. For the times before timezones were invented (late 1800s?) we can safely assume that fold=0 always. To be safe, I would set the fold cut-off at the year 1000. |
I encountered this issue in Python 3.8. I consider it to be a bug because it's an instance of a class-defined constant (datetime.min) not working with a method of that class (timestamp) when all other values that the class could take work with the method AND the constant works with all of the class' other methods. And it has practical implications: As belopolsky said above, "a reasonable design can use datetime.min/max as placeholders for unknown times far in the past/future compensating for the lack [of] datetime ±inf." Since datetime.min lies so close to the edges of datetime's value-space, perhaps it should be made offset-aware and placed in UTC so that it stops breaking timestamp() when the user is in the wrong timezone. Alternatively, there could be a note in the documentation for datetime.timestamp() about this edge case. Assuming similarly bizarre behavior happens at datetime.max for one or more datetime methods (and for consistency's sake), it should probably be put in UTC as well. |
Perhaps min and max should be moved 24 hours up and down respectively? This would make their names more accurately reflect valid range of the class and also allow them to be used as effective -inf +inf. |
This just bit me and from a user experience perspective, I agree that it seems like a bug. At a minimum, it would probably make sense for
|
This is also relevant for the Python implementation. Error message has improved too, There was no real conclusion previously, what do we want to do? |
The problem isn't really even in datetime; Line 1991 in 563ab5c
This returns a |
This is a bit out of topic, but could some kind soul recommend me some way to "sanitize" invalid parameters? I'm trying to parse the |
Note: these values reflect the state of the issue at the time it was migrated and might not reflect the current state.
Show more details
GitHub fields:
bugs.python.org fields:
The text was updated successfully, but these errors were encountered: