|
1 |
| -import datetime |
2 |
| -import re |
| 1 | +from __future__ import annotations |
3 | 2 |
|
| 3 | +import datetime |
4 | 4 | from enum import auto
|
| 5 | +import re |
| 6 | +from typing import TYPE_CHECKING |
5 | 7 |
|
| 8 | +if TYPE_CHECKING: |
| 9 | + from undate.interval import UndateInterval |
6 | 10 | try:
|
7 | 11 | # StrEnum was only added in python 3.11
|
8 | 12 | from enum import StrEnum
|
|
14 | 18 | from typing import Dict, Optional, Union
|
15 | 19 |
|
16 | 20 | from undate.converters.base import BaseDateConverter
|
17 |
| -from undate.date import ONE_DAY, ONE_MONTH_MAX, ONE_YEAR, Date, DatePrecision, Timedelta |
| 21 | +from undate.date import ONE_DAY, ONE_MONTH_MAX, Date, DatePrecision, Timedelta |
18 | 22 |
|
19 | 23 |
|
20 | 24 | class Calendar(StrEnum):
|
@@ -218,7 +222,7 @@ def __repr__(self) -> str:
|
218 | 222 | return f"<Undate{label_str} {self} ({self.calendar.name.title()})>"
|
219 | 223 |
|
220 | 224 | @classmethod
|
221 |
| - def parse(cls, date_string, format) -> Union["Undate", "UndateInterval"]: |
| 225 | + def parse(cls, date_string, format) -> Union["Undate", UndateInterval]: |
222 | 226 | """parse a string to an undate or undate interval using the specified format;
|
223 | 227 | for now, only supports named converters"""
|
224 | 228 | converter_cls = BaseDateConverter.available_converters().get(format, None)
|
@@ -487,98 +491,3 @@ def _missing_digit_minmax(
|
487 | 491 | min_val = int("".join(new_min_val))
|
488 | 492 | max_val = int("".join(new_max_val))
|
489 | 493 | return (min_val, max_val)
|
490 |
| - |
491 |
| - |
492 |
| -class UndateInterval: |
493 |
| - """A date range between two uncertain dates. |
494 |
| -
|
495 |
| - :param earliest: Earliest undate |
496 |
| - :type earliest: `undate.Undate` |
497 |
| - :param latest: Latest undate |
498 |
| - :type latest: `undate.Undate` |
499 |
| - :param label: A string to label a specific undate interval, similar to labels of `undate.Undate`. |
500 |
| - :type label: `str` |
501 |
| - """ |
502 |
| - |
503 |
| - # date range between two undates |
504 |
| - earliest: Union[Undate, None] |
505 |
| - latest: Union[Undate, None] |
506 |
| - label: Union[str, None] |
507 |
| - |
508 |
| - # TODO: let's think about adding an optional precision / length /size field |
509 |
| - # using DatePrecision |
510 |
| - |
511 |
| - def __init__( |
512 |
| - self, |
513 |
| - earliest: Optional[Undate] = None, |
514 |
| - latest: Optional[Undate] = None, |
515 |
| - label: Optional[str] = None, |
516 |
| - ): |
517 |
| - # for now, assume takes two undate objects |
518 |
| - self.earliest = earliest |
519 |
| - self.latest = latest |
520 |
| - self.label = label |
521 |
| - |
522 |
| - def __str__(self) -> str: |
523 |
| - # using EDTF syntax for open ranges |
524 |
| - return "%s/%s" % (self.earliest or "..", self.latest or "") |
525 |
| - |
526 |
| - def format(self, format) -> str: |
527 |
| - """format this undate interval as a string using the specified format; |
528 |
| - for now, only supports named converters""" |
529 |
| - converter_cls = BaseDateConverter.available_converters().get(format, None) |
530 |
| - if converter_cls: |
531 |
| - return converter_cls().to_string(self) |
532 |
| - |
533 |
| - raise ValueError(f"Unsupported format '{format}'") |
534 |
| - |
535 |
| - def __repr__(self) -> str: |
536 |
| - if self.label: |
537 |
| - return "<UndateInterval '%s' (%s)>" % (self.label, self) |
538 |
| - return "<UndateInterval %s>" % self |
539 |
| - |
540 |
| - def __eq__(self, other) -> bool: |
541 |
| - # consider interval equal if both dates are equal |
542 |
| - return self.earliest == other.earliest and self.latest == other.latest |
543 |
| - |
544 |
| - def duration(self) -> Timedelta: |
545 |
| - """Calculate the duration between two undates. |
546 |
| - Note that durations are inclusive (i.e., a closed interval), and |
547 |
| - include both the earliest and latest date rather than the difference |
548 |
| - between them. |
549 |
| -
|
550 |
| - :returns: A duration |
551 |
| - :rtype: Timedelta |
552 |
| - """ |
553 |
| - # what is the duration of this date range? |
554 |
| - |
555 |
| - # if range is open-ended, can't calculate |
556 |
| - if self.earliest is None or self.latest is None: |
557 |
| - return NotImplemented |
558 |
| - |
559 |
| - # if both years are known, subtract end of range from beginning of start |
560 |
| - if self.latest.known_year and self.earliest.known_year: |
561 |
| - return self.latest.latest - self.earliest.earliest + ONE_DAY |
562 |
| - |
563 |
| - # if neither year is known... |
564 |
| - elif not self.latest.known_year and not self.earliest.known_year: |
565 |
| - # under what circumstances can we assume that if both years |
566 |
| - # are unknown the dates are in the same year or sequential? |
567 |
| - duration = self.latest.earliest - self.earliest.earliest |
568 |
| - # if we get a negative, we've wrapped from end of one year |
569 |
| - # to the beginning of the next; |
570 |
| - # recalculate assuming second date is in the subsequent year |
571 |
| - if duration.days < 0: |
572 |
| - end = self.latest.earliest + ONE_YEAR |
573 |
| - duration = end - self.earliest.earliest |
574 |
| - |
575 |
| - # add the additional day *after* checking for a negative |
576 |
| - # or after recalculating with adjusted year |
577 |
| - duration += ONE_DAY |
578 |
| - |
579 |
| - return duration |
580 |
| - |
581 |
| - else: |
582 |
| - # is there any meaningful way to calculate duration |
583 |
| - # if one year is known and the other is not? |
584 |
| - raise NotImplementedError |
0 commit comments