Search code examples
pythondatetimetimedelta

How to convert a python datetime.timedelta to years?


I need to check if n years have been since some date. I'm using timedelta from the datetime module and I don't know how to convert it to years.


Solution

  • You need more than a timedelta to tell how many years have passed; you also need to know the beginning (or ending) date. (It's a leap year thing.)

    Your best bet is to use the dateutil.relativedelta object, but that's a 3rd party module. If you want to know the datetime that was n years from some date (defaulting to right now), you can do the following::

    from dateutil.relativedelta import relativedelta
    
    def yearsago(years, from_date=None):
        if from_date is None:
            from_date = datetime.now()
        return from_date - relativedelta(years=years)
    

    If you'd rather stick with the standard library, the answer is a little more complex::

    from datetime import datetime
    def yearsago(years, from_date=None):
        if from_date is None:
            from_date = datetime.now()
        try:
            return from_date.replace(year=from_date.year - years)
        except ValueError:
            # Must be 2/29!
            assert from_date.month == 2 and from_date.day == 29 # can be removed
            return from_date.replace(month=2, day=28,
                                     year=from_date.year-years)
    

    If it's 2/29, and 18 years ago there was no 2/29, this function will return 2/28. If you'd rather return 3/1, just change the last return statement to read::

        return from_date.replace(month=3, day=1,
                                 year=from_date.year-years)
    

    Your question originally said you wanted to know how many years it's been since some date. Assuming you want an integer number of years, you can guess based on 365.2425 days per year and then check using either of the yearsago functions defined above::

    def num_years(begin, end=None):
        if end is None:
            end = datetime.now()
        num_years = int((end - begin).days / 365.2425)
        if begin > yearsago(num_years, end):
            return num_years - 1
        else:
            return num_years