Skip to content
Merged
Show file tree
Hide file tree
Changes from 19 commits
Commits
Show all changes
22 commits
Select commit Hold shift + click to select a range
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
15 changes: 15 additions & 0 deletions .github/workflows/test.yml
Original file line number Diff line number Diff line change
Expand Up @@ -118,6 +118,10 @@ jobs:
os: windows-latest
tox_env: "py313"

- name: "windows-py314"
python: "3.14"
os: windows-latest
tox_env: "py314"

- name: "ubuntu-py39-lsof-numpy-pexpect"
python: "3.9"
Expand Down Expand Up @@ -163,6 +167,12 @@ jobs:
tox_env: "py313-pexpect"
use_coverage: true

- name: "ubuntu-py314"
python: "3.14"
os: ubuntu-latest
tox_env: "py314"
use_coverage: true

- name: "ubuntu-pypy3-xdist"
python: "pypy-3.9"
os: ubuntu-latest
Expand Down Expand Up @@ -190,6 +200,10 @@ jobs:
os: macos-latest
tox_env: "py313-xdist"

- name: "macos-py314"
python: "3.14"
os: macos-latest
tox_env: "py314-xdist"

- name: "plugins"
python: "3.12"
Expand Down Expand Up @@ -240,6 +254,7 @@ jobs:
with:
python-version: ${{ matrix.python }}
check-latest: ${{ endsWith(matrix.python, '-dev') }}
Copy link
Member

Choose a reason for hiding this comment

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

for check-latest to work the matrix.python entries should be named 3.14-dev

Copy link
Member Author

Choose a reason for hiding this comment

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

Why not always have check-latest: true?

Copy link
Member

@jakkdl jakkdl May 30, 2025

Choose a reason for hiding this comment

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

https://github.com/actions/setup-python/blob/main/docs/advanced-usage.md#check-latest-version

apparently performance & stability, though kind of seems more reliable to have it set to true to ensure you don't randomly get an old cached version. No clue what the performance hit would be

Copy link
Member Author

Choose a reason for hiding this comment

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

I prefer not to use -dev because then you have to remember to remove it for final release. I prefer allow-prereleases

Copy link
Member Author

Choose a reason for hiding this comment

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

Suggested change
check-latest: ${{ endsWith(matrix.python, '-dev') }}
check-latest: true

Copy link
Member

Choose a reason for hiding this comment

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

seems fine 👍

allow-prereleases: true

- name: Install dependencies
run: |
Expand Down
1 change: 1 addition & 0 deletions changelog/13308.improvement.rst
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
Added official support for Python 3.14.
3 changes: 2 additions & 1 deletion pyproject.toml
Original file line number Diff line number Diff line change
Expand Up @@ -38,6 +38,7 @@ classifiers = [
"Programming Language :: Python :: 3.11",
"Programming Language :: Python :: 3.12",
"Programming Language :: Python :: 3.13",
"Programming Language :: Python :: 3.14",
"Topic :: Software Development :: Libraries",
"Topic :: Software Development :: Testing",
"Topic :: Utilities",
Expand Down Expand Up @@ -349,7 +350,7 @@ ignore = "W009"

[tool.pyproject-fmt]
indent = 4
max_supported_python = "3.13"
max_supported_python = "3.14"

[tool.pytest.ini_options]
minversion = "2.0"
Expand Down
8 changes: 6 additions & 2 deletions src/_pytest/pastebin.py
Original file line number Diff line number Diff line change
Expand Up @@ -76,6 +76,7 @@ def create_new_paste(contents: str | bytes) -> str:
:returns: URL to the pasted contents, or an error message.
"""
import re
from urllib.error import HTTPError
from urllib.parse import urlencode
from urllib.request import urlopen

Expand All @@ -85,8 +86,11 @@ def create_new_paste(contents: str | bytes) -> str:
response: str = (
urlopen(url, data=urlencode(params).encode("ascii")).read().decode("utf-8")
)
except OSError as exc_info: # urllib errors
return f"bad response: {exc_info}"
except HTTPError as e: # urllib.error errors
with e: # HTTPErrors are also http responses that must be closed!
return f"bad response: {e}"
except OSError as e: # urllib errors
return f"bad response: {e}"
m = re.search(r'href="/raw/(\w+)"', response)
if m:
return f"{url}/show/{m.group(1)}"
Expand Down
9 changes: 7 additions & 2 deletions testing/python/raises.py
Original file line number Diff line number Diff line change
@@ -1,6 +1,7 @@
# mypy: allow-untyped-defs
from __future__ import annotations

import io
import re
import sys

Expand Down Expand Up @@ -399,7 +400,11 @@ def test_issue_11872(self) -> None:

https://github.com/python/cpython/issues/98778
"""
from email.message import Message
from urllib.error import HTTPError

with pytest.raises(HTTPError, match="Not Found"):
raise HTTPError(code=404, msg="Not Found", fp=None, hdrs=None, url="") # type: ignore [arg-type]
with pytest.raises(HTTPError, match="Not Found") as exc_info:
raise HTTPError(
code=404, msg="Not Found", fp=io.BytesIO(), hdrs=Message(), url=""
)
exc_info.value.close() # avoid a resource warning
5 changes: 5 additions & 0 deletions testing/test_debugging.py
Original file line number Diff line number Diff line change
Expand Up @@ -1373,6 +1373,10 @@ def do_quit(self, *args):
result.stdout.fnmatch_lines(["*runcall_called*", "* 1 passed in *"])


@pytest.mark.xfail(
sys.version_info >= (3, 14),
reason="C-D now quits the test session, rather than failing the test. See https://github.com/python/cpython/issues/124703",
)
def test_raises_bdbquit_with_eoferror(pytester: Pytester) -> None:
"""It is not guaranteed that DontReadFromInput's read is called."""
p1 = pytester.makepyfile(
Expand All @@ -1387,6 +1391,7 @@ def test(monkeypatch):
"""
)
result = pytester.runpytest(str(p1))
result.assert_outcomes(failed=1)
result.stdout.fnmatch_lines(["E *BdbQuit", "*= 1 failed in*"])
assert result.ret == 1

Expand Down
2 changes: 1 addition & 1 deletion testing/test_parseopt.py
Original file line number Diff line number Diff line change
Expand Up @@ -28,7 +28,7 @@ def test_no_help_by_default(self) -> None:

def test_custom_prog(self, parser: parseopt.Parser) -> None:
"""Custom prog can be set for `argparse.ArgumentParser`."""
assert parser._getparser().prog == os.path.basename(sys.argv[0])
assert parser._getparser().prog == argparse.ArgumentParser().prog
parser.prog = "custom-prog"
assert parser._getparser().prog == "custom-prog"

Expand Down
33 changes: 24 additions & 9 deletions testing/test_unraisableexception.py
Original file line number Diff line number Diff line change
Expand Up @@ -12,6 +12,24 @@

PYPY = hasattr(sys, "pypy_version_info")

UNRAISABLE_LINE = (
(
" * PytestUnraisableExceptionWarning: Exception ignored while calling "
"deallocator <function BrokenDel.__del__ at *>: None"
)
if sys.version_info >= (3, 14)
else " * PytestUnraisableExceptionWarning: Exception ignored in: <function BrokenDel.__del__ at *>"
)

TRACEMALLOC_LINES = (
()
if sys.version_info >= (3, 14)
else (
" Enable tracemalloc to get traceback where the object was allocated.",
" See https* for more info.",
)
)


@pytest.mark.skipif(PYPY, reason="garbage-collection differences make this flaky")
@pytest.mark.filterwarnings("default::pytest.PytestUnraisableExceptionWarning")
Expand All @@ -36,13 +54,12 @@ def test_2(): pass
[
"*= warnings summary =*",
"test_it.py::test_it",
" * PytestUnraisableExceptionWarning: Exception ignored in: <function BrokenDel.__del__ at *>",
UNRAISABLE_LINE,
" ",
" Traceback (most recent call last):",
" ValueError: del is broken",
" ",
" Enable tracemalloc to get traceback where the object was allocated.",
" See https* for more info.",
*TRACEMALLOC_LINES,
" warnings.warn(pytest.PytestUnraisableExceptionWarning(msg))",
]
)
Expand Down Expand Up @@ -75,13 +92,12 @@ def test_2(): pass
[
"*= warnings summary =*",
"test_it.py::test_it",
" * PytestUnraisableExceptionWarning: Exception ignored in: <function BrokenDel.__del__ at *>",
UNRAISABLE_LINE,
" ",
" Traceback (most recent call last):",
" ValueError: del is broken",
" ",
" Enable tracemalloc to get traceback where the object was allocated.",
" See https* for more info.",
*TRACEMALLOC_LINES,
" warnings.warn(pytest.PytestUnraisableExceptionWarning(msg))",
]
)
Expand Down Expand Up @@ -115,13 +131,12 @@ def test_2(): pass
[
"*= warnings summary =*",
"test_it.py::test_it",
" * PytestUnraisableExceptionWarning: Exception ignored in: <function BrokenDel.__del__ at *>",
UNRAISABLE_LINE,
" ",
" Traceback (most recent call last):",
" ValueError: del is broken",
" ",
" Enable tracemalloc to get traceback where the object was allocated.",
" See https* for more info.",
*TRACEMALLOC_LINES,
" warnings.warn(pytest.PytestUnraisableExceptionWarning(msg))",
]
)
Expand Down
1 change: 1 addition & 0 deletions tox.ini
Original file line number Diff line number Diff line change
Expand Up @@ -9,6 +9,7 @@ envlist =
py311
py312
py313
py314
pypy3
py39-{pexpect,xdist,unittestextras,numpy,pluggymain,pylib}
doctesting
Expand Down