Description
This SPEC recommends that all projects across the Scientific Python ecosystem adopt a common time-based policy for dropping support of older Python and core package versions.
All versions refer to feature releases (i.e., Python 3.8.0, NumPy 1.19.0; not Python 3.8.1, NumPy 1.19.2).
Specifically, we recommend that:
- Support for a given version of Python be dropped 3 years after its initial release.
- Support for a given version of other core packages be dropped 2 years after their initial release.
Motivation
Limiting the scope of supported dependencies is an effective way for packages to limit maintenance burden. Combinations of packages need to be tested, which impacts also on continuous integration times and infrastructure upkeep. Code itself also becomes more complicated when it has to be aware of various combinations of configurations.
Adoption of this SPEC will ensure a consistent support policy across packages, and reduce the need for individual projects to divise similar policies.
Ultimately, reduced maintenance burden frees up developer time, which translates into more features, bugfixes, and optimizations for users.
Background
In the past, longer support cycles were common. There were several reasons for this, including the Python 2 / 3 transition, difficulties installing packages, and users needing to use old, operating-system provided versions of Python. The situation has since improved due to improved installations via binary wheels, virtual environments becoming commonplace, and support for Python 2 being dropped.
Drop Schedule
2023 – Quarter 3:
- 02 Jul 2023: drop pandas 1.3.0 (initially released on Jul 02, 2021)
- 08 Jul 2023: drop networkx 2.6 (initially released on Jul 08, 2021)
- 23 Jul 2023: drop xarray 0.19.0 (initially released on Jul 23, 2021)
- 01 Aug 2023: drop ipython 7.26.0 (initially released on Aug 01, 2021)
- 27 Aug 2023: drop ipython 7.27.0 (initially released on Aug 27, 2021)
- 24 Sep 2023: drop scikit-learn 1.0 (initially released on Sep 24, 2021)
- 25 Sep 2023: drop ipython 7.28.0 (initially released on Sep 25, 2021)
2023 – Quarter 4:
- 05 Oct 2023: drop python 3.9 (initially released on Oct 05, 2020)
- 30 Oct 2023: drop ipython 7.29.0 (initially released on Oct 30, 2021)
- 02 Nov 2023: drop xarray 0.20.0 (initially released on Nov 02, 2021)
- 16 Nov 2023: drop matplotlib 3.5.0 (initially released on Nov 16, 2021)
- 26 Nov 2023: drop ipython 7.30.0 (initially released on Nov 26, 2021)
- 03 Dec 2023: drop scikit-image 0.19.0 (initially released on Dec 03, 2021)
- 31 Dec 2023: drop numpy 1.22.0 (initially released on Dec 31, 2021)
2024 – Quarter 1:
- 05 Jan 2024: drop ipython 7.31.0 (initially released on Jan 05, 2022)
- 12 Jan 2024: drop ipython 8.0.0 (initially released on Jan 12, 2022)
- 22 Jan 2024: drop pandas 1.4.0 (initially released on Jan 22, 2022)
- 28 Jan 2024: drop xarray 0.21.0 (initially released on Jan 28, 2022)
- 05 Feb 2024: drop scipy 1.8.0 (initially released on Feb 05, 2022)
- 25 Feb 2024: drop ipython 7.32.0 (initially released on Feb 25, 2022)
- 25 Feb 2024: drop ipython 8.1.0 (initially released on Feb 25, 2022)
- 28 Feb 2024: drop networkx 2.7 (initially released on Feb 28, 2022)
- 01 Mar 2024: drop xarray 2022.3.0 (initially released on Mar 02, 2022)
- 26 Mar 2024: drop ipython 8.2.0 (initially released on Mar 27, 2022)
2024 – Quarter 2:
- 08 Apr 2024: drop networkx 2.8 (initially released on Apr 09, 2022)
- 28 Apr 2024: drop ipython 7.33.0 (initially released on Apr 29, 2022)
- 28 Apr 2024: drop ipython 8.3.0 (initially released on Apr 29, 2022)
- 11 May 2024: drop scikit-learn 1.1.0 (initially released on May 12, 2022)
- 27 May 2024: drop ipython 7.34.0 (initially released on May 28, 2022)
- 27 May 2024: drop ipython 8.4.0 (initially released on May 28, 2022)
- 21 Jun 2024: drop numpy 1.23.0 (initially released on Jun 22, 2022)
2024 – Quarter 3:
- 21 Jul 2024: drop xarray 2022.6.0 (initially released on Jul 22, 2022)
- 28 Jul 2024: drop scipy 1.9.0 (initially released on Jul 29, 2022)
- 05 Sep 2024: drop ipython 8.5.0 (initially released on Sep 06, 2022)
- 15 Sep 2024: drop matplotlib 3.6.0 (initially released on Sep 16, 2022)
- 18 Sep 2024: drop pandas 1.5.0 (initially released on Sep 19, 2022)
- 28 Sep 2024: drop xarray 2022.9.0 (initially released on Sep 29, 2022)
2024 – Quarter 4:
- 03 Oct 2024: drop python 3.10 (initially released on Oct 04, 2021)
- 12 Oct 2024: drop xarray 2022.10.0 (initially released on Oct 13, 2022)
- 29 Oct 2024: drop ipython 8.6.0 (initially released on Oct 30, 2022)
- 03 Nov 2024: drop xarray 2022.11.0 (initially released on Nov 04, 2022)
- 27 Nov 2024: drop ipython 8.7.0 (initially released on Nov 28, 2022)
- 01 Dec 2024: drop xarray 2022.12.0 (initially released on Dec 02, 2022)
- 07 Dec 2024: drop scikit-learn 1.2.0 (initially released on Dec 08, 2022)
- 17 Dec 2024: drop numpy 1.24.0 (initially released on Dec 18, 2022)
2025 – Quarter 1:
- 02 Jan 2025: drop ipython 8.8.0 (initially released on Jan 03, 2023)
- 02 Jan 2025: drop scipy 1.10.0 (initially released on Jan 03, 2023)
- 07 Jan 2025: drop networkx 3.0 (initially released on Jan 08, 2023)
- 17 Jan 2025: drop xarray 2023.1.0 (initially released on Jan 18, 2023)
- 26 Jan 2025: drop ipython 8.9.0 (initially released on Jan 27, 2023)
- 06 Feb 2025: drop xarray 2023.2.0 (initially released on Feb 07, 2023)
- 09 Feb 2025: drop ipython 8.10.0 (initially released on Feb 10, 2023)
- 12 Feb 2025: drop matplotlib 3.7.0 (initially released on Feb 13, 2023)
- 27 Feb 2025: drop ipython 8.11.0 (initially released on Feb 28, 2023)
- 27 Feb 2025: drop scikit-image 0.20.0 (initially released on Feb 28, 2023)
- 21 Mar 2025: drop xarray 2023.3.0 (initially released on Mar 22, 2023)
- 29 Mar 2025: drop ipython 8.12.0 (initially released on Mar 30, 2023)
2025 – Quarter 2:
- 02 Apr 2025: drop pandas 2.0.0 (initially released on Apr 03, 2023)
- 03 Apr 2025: drop networkx 3.1 (initially released on Apr 04, 2023)
- 13 Apr 2025: drop xarray 2023.4.0 (initially released on Apr 14, 2023)
- 27 Apr 2025: drop ipython 8.13.0 (initially released on Apr 28, 2023)
- 18 May 2025: drop xarray 2023.5.0 (initially released on May 19, 2023)
- 01 Jun 2025: drop scikit-image 0.21.0 (initially released on Jun 02, 2023)
- 01 Jun 2025: drop ipython 8.14.0 (initially released on Jun 02, 2023)
- 16 Jun 2025: drop numpy 1.25.0 (initially released on Jun 17, 2023)
- 22 Jun 2025: drop xarray 2023.6.0 (initially released on Jun 23, 2023)
- 24 Jun 2025: drop scipy 1.11.0 (initially released on Jun 25, 2023)
2025 – Quarter 4:
- 23 Oct 2025: drop python 3.11 (initially released on Oct 24, 2022)
2026 – Quarter 4:
- 01 Oct 2026: drop python 3.12 (initially released on Oct 02, 2023)
Implementation
Notes
-
This document builds on NEP 29, which describes several alternatives including ad hoc version support, all CPython supported versions, default version on Linux distribution, N minor versions of Python, and time window from the X.Y.1 Python release.
-
Code to generate support and drop schedule tables:
import requests
import itertools
import collections
from datetime import datetime, timedelta
import pandas as pd
from packaging.version import Version
py_releases = {
"3.8": "Oct 14, 2019",
"3.9": "Oct 5, 2020",
"3.10": "Oct 4, 2021",
"3.11": "Oct 24, 2022",
"3.12": "Oct 2, 2023",
}
core_packages = [
# Path(x).stem for x in glob("../core-projects/*.md") if "_index" not in x
"numpy",
"scipy",
"matplotlib",
"pandas",
"scikit-image",
"networkx",
"scikit-learn",
"xarray",
"ipython",
]
plus36 = timedelta(days=int(365 * 3))
plus24 = timedelta(days=int(365 * 2))
# Release data
now = datetime.now()
def get_release_dates(package, support_time=plus24):
releases = {}
print(f"Querying pypi.org for {package} versions...", end="", flush=True)
response = requests.get(
f"https://pypi.org/simple/{package}",
headers={"Accept": "application/vnd.pypi.simple.v1+json"},
).json()
print("OK")
file_date = collections.defaultdict(list)
for f in response["files"]:
ver = f["filename"].split("-")[1]
try:
version = Version(ver)
except:
continue
if version.is_prerelease or version.micro != 0:
continue
release_date = None
for format in ["%Y-%m-%dT%H:%M:%S.%fZ", "%Y-%m-%dT%H:%M:%SZ"]:
try:
release_date = datetime.strptime(f["upload-time"], format)
except:
pass
if not release_date:
continue
file_date[version].append(release_date)
release_date = {v: min(file_date[v]) for v in file_date}
for ver, release_date in sorted(release_date.items()):
drop_date = release_date + support_time
if drop_date >= datetime.now():
releases[ver] = {
"release_date": release_date,
"drop_date": drop_date,
}
return releases
package_releases = {
"python": {
version: {
"release_date": datetime.strptime(release_date, "%b %d, %Y"),
"drop_date": datetime.strptime(release_date, "%b %d, %Y") + plus36,
}
for version, release_date in py_releases.items()
}
}
package_releases |= {package: get_release_dates(package) for package in core_packages}
# filter all items whose drop_date are in the past
package_releases = {
package: {
version: dates
for version, dates in releases.items()
if dates["drop_date"] > now
}
for package, releases in package_releases.items()
}
# Save Gantt chart
print("Saving Mermaid chart to chart.md")
with open("chart.md", "w") as fh:
fh.write(
"""gantt
dateFormat YYYY-MM-DD
axisFormat %m / %Y
title Support Window"""
)
for name, releases in package_releases.items():
fh.write(f"\n\nsection {name}")
for version, dates in releases.items():
fh.write(
f"\n{version} : {dates['release_date'].strftime('%Y-%m-%d')},{dates['drop_date'].strftime('%Y-%m-%d')}"
)
fh.write("\n")
# Print drop schedule
rel = {}
for name, releases in package_releases.items():
rel |= {
" ".join([name, str(ver)]): [dates["release_date"], dates["drop_date"]]
for ver, dates in releases.items()
}
print("Saving drop schedule to schedule.md")
with open("schedule.md", "w") as fh:
current_quarter = None
# Sort by drop date
rel = dict(sorted(rel.items(), key=lambda item: item[1][1]))
for package, dates in rel.items():
qt = pd.to_datetime(dates[1]).to_period("Q")
# If drop date is in a new quarter, write out a heading
if qt != current_quarter:
if current_quarter != None:
fh.write("\n")
fh.write(f'{str(qt).replace("Q", " – Quarter ")}:\n\n')
current_quarter = qt
fh.write(
f"- {dates[1].strftime('%d %b %Y')}: drop {package} (initially released on {dates[0].strftime('%b %d, %Y')})\n"
)