Skip to content

Commit 3dbc352

Browse files
authoredFeb 16, 2025··
Add support for Python 3.13 and drop support for Python 3.10 (#784)
* Update to python 3.13 + updated pyproject.toml format * Update foramtting (safe fixes) * Update foramtting (less safe fixes) * Updated changelog
1 parent f7a141d commit 3dbc352

20 files changed

+86
-104
lines changed
 

‎.github/workflows/benchmarks.yml

+1-1
Original file line numberDiff line numberDiff line change
@@ -22,7 +22,7 @@ jobs:
2222
- name: Set up Python
2323
uses: actions/setup-python@v4
2424
with:
25-
python-version: '3.10'
25+
python-version: '3.11'
2626
cache: 'poetry'
2727
- name: Install project
2828
run: poetry install -E all

‎.github/workflows/python-lint-tests.yml

+1-1
Original file line numberDiff line numberDiff line change
@@ -17,7 +17,7 @@ jobs:
1717
strategy:
1818
fail-fast: true
1919
matrix:
20-
python-version: ['3.10', '3.11', '3.12']
20+
python-version: ['3.11', '3.11', '3.13']
2121
os: [ubuntu-latest, macos-latest, windows-latest]
2222
defaults:
2323
run:

‎.readthedocs.yml

+1-1
Original file line numberDiff line numberDiff line change
@@ -13,7 +13,7 @@ sphinx:
1313
build:
1414
os: ubuntu-22.04
1515
tools:
16-
python: "3.11"
16+
python: "3.13"
1717
jobs:
1818
pre_create_environment:
1919
- pip install poetry

‎CHANGELOG.md

+1
Original file line numberDiff line numberDiff line change
@@ -6,6 +6,7 @@ Release date: UNRELEASED
66

77
### New features and improvements
88

9+
* Added support for Python 3.13 and dropped support for Python 3.10 (#784)
910
* Added a `--orientation` option to the `pagerotate` command to conditionally rotate the page to a target orientation (thanks to @gatesphere) (#705)
1011
* Added a `lineshuffle` command to randomize the plotting order of lines in the current geometry (thanks to @gatesphere) (#715)
1112

‎README.md

+1-1
Original file line numberDiff line numberDiff line change
@@ -189,7 +189,7 @@ curvy (circles, bezier curves, etc.) to lines made of small segments. _vpype_ do
189189
Detailed installation instructions are available in the [latest documentation](https://vpype.readthedocs.io/en/latest/install.html).
190190

191191
TL;DR:
192-
- Python 3.12 is recommended, but *vpype* is also compatible with Python 3.10 and 3.11.
192+
- Python 3.13 is recommended, but *vpype* is also compatible with Python 3.11 and 3.12.
193193
- *vpype* is published on the [Python Package Index](https://pypi.org) and can be installed using [pipx](https://pypa.github.io/pipx/):
194194
```bash
195195
pipx install "vpype[all]"

‎docs/install.rst

+2-2
Original file line numberDiff line numberDiff line change
@@ -9,7 +9,7 @@ This page explain how to install *vpype* for end-users. If you intend to develop
99

1010
.. note::
1111

12-
The recommended Python version is 3.12.1 or later. *vpype* is also compatible with Python 3.10 and 3.11.
12+
The recommended Python version is 3.13. *vpype* is also compatible with Python 3.11 and 3.12.
1313

1414
.. warning::
1515

@@ -38,7 +38,7 @@ You can ensure that the installed Python interpreter is properly installed by r
3838

3939
It should produce an output similar to::
4040

41-
Python 3.12.1
41+
Python 3.13.2
4242

4343
The version number should match the installer you used.
4444

‎poetry.lock

+3-28
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

‎pyproject.toml

+48-40
Original file line numberDiff line numberDiff line change
@@ -1,18 +1,49 @@
1-
[tool.poetry]
1+
[project]
22
name = "vpype"
33
version = "1.15.0a0"
44
description = "The Swiss Army knife of vector graphics for pen plotters"
5-
authors = ["Antoine Beyeler <abeyeler@ab-ware.com>"]
5+
authors = [
6+
{ name = "Antoine Beyeler", email = "abeyeler@ab-ware.com>" }
7+
]
68
license = "MIT"
79
readme = "README.md"
8-
homepage = "https://github.com/abey79/vpype"
9-
documentation = "https://vpype.readthedocs.io/en/latest/"
10-
classifiers = [
11-
"Development Status :: 5 - Production/Stable",
12-
"Environment :: Console",
13-
"Topic :: Artistic Software",
14-
"Topic :: Multimedia :: Graphics",
10+
requires-python = ">=3.11, <3.14"
11+
dynamic = ["classifiers"]
12+
13+
dependencies = [
14+
"asteval>=0.9.26",
15+
"cachetools>=4.2.2",
16+
"click>=8.0.1,<8.2.0",
17+
"multiprocess>=0.70.11",
18+
"numpy>=1.25,<3",
19+
"pnoise>=0.2.0",
20+
"pyphen>=0.14,<0.16",
21+
"scipy>=1.6",
22+
"Shapely>=1.8.2",
23+
"svgelements>=1.6.10",
24+
"svgwrite~=1.4",
25+
"tomli>=2.0.0",
26+
]
27+
28+
[project.optional-dependencies]
29+
all = [
30+
"matplotlib>=3.3.2",
31+
"glcontext>=2.3.2", # 2.3.2 needed to fix #200
32+
"moderngl>=5.6.2,!=5.7.1,!=5.7.2", # see moderngl/moderngl#525
33+
"Pillow>=9.0.1",
34+
"PySide6>=6.4.0.1,!=6.6.2",
1535
]
36+
37+
38+
[project.urls]
39+
documentation = "https://vpype.readthedocs.io/en/latest/"
40+
repository = "https://github.com/abey79/vpype"
41+
42+
[project.scripts]
43+
vpype = "vpype_cli.cli:cli"
44+
45+
46+
[tool.poetry]
1647
packages = [
1748
{ include = "vpype" },
1849
{ include = "vpype_cli" },
@@ -29,31 +60,14 @@ include = [
2960
"vpype_viewer/qtviewer/resources/*",
3061
]
3162

32-
[tool.poetry.scripts]
33-
vpype = "vpype_cli.cli:cli"
63+
# Poetry autofills license and Python version related items.
64+
classifiers = [
65+
"Development Status :: 5 - Production/Stable",
66+
"Environment :: Console",
67+
"Topic :: Artistic Software",
68+
"Topic :: Multimedia :: Graphics",
69+
]
3470

35-
[tool.poetry.dependencies]
36-
python = ">=3.10, <3.13"
37-
38-
asteval = ">=0.9.26"
39-
cachetools = ">=4.2.2"
40-
click = ">=8.0.1,<8.2.0"
41-
multiprocess = ">=0.70.11"
42-
numpy = ">=1.25,<3"
43-
pnoise = ">=0.2.0"
44-
pyphen = ">=0.14,<0.16"
45-
scipy = ">=1.6"
46-
Shapely = ">=1.8.2"
47-
svgelements = ">=1.6.10"
48-
svgwrite = "~1.4"
49-
tomli = ">=2.0.0"
50-
51-
# additional dependencies for the viewer and the `show` command
52-
matplotlib = { version = ">=3.3.2", optional = true }
53-
glcontext = { version = ">=2.3.2", optional = true } # 2.3.2 needed to fix #200
54-
moderngl = { version = ">=5.6.2,!=5.7.1,!=5.7.2", optional = true } # see moderngl/moderngl#525
55-
Pillow = { version = ">=9.0.1", optional = true }
56-
PySide6 = { version = ">=6.4.0.1,!=6.6.2", optional = true }
5771

5872
[tool.poetry.group.dev.dependencies]
5973
coverage = {extras = ["toml"], version = ">=5.4"}
@@ -83,9 +97,6 @@ sphinx-click = ">=4.3.0"
8397
sphinx-copybutton = ">=0.5.0"
8498

8599

86-
[tool.poetry.extras]
87-
all = ["matplotlib", "glcontext", "moderngl", "Pillow", "PySide6"]
88-
89100
[build-system]
90101
requires = ["poetry-core>=1.0.8"]
91102
build-backend = "poetry.core.masonry.api"
@@ -129,13 +140,10 @@ exclude_lines = [
129140
'@(abc\.)?abstractmethod',
130141
]
131142

132-
[tool.black]
133-
line-length = 95
134-
target-version = ["py39", "py310", "py311"]
135143

136144
[tool.ruff]
137145
line-length = 95
138-
target-version = "py39"
146+
target-version = "py311"
139147
exclude = ["examples", "scripts"]
140148

141149
[tool.ruff.lint]

‎tests/conftest.py

+2-1
Original file line numberDiff line numberDiff line change
@@ -7,7 +7,8 @@
77
import pathlib
88
import random
99
import string
10-
from typing import Callable, Protocol
10+
from collections.abc import Callable
11+
from typing import Protocol
1112
from xml.dom import minidom
1213
from xml.etree import ElementTree
1314

‎vpype/io.py

+5-5
Original file line numberDiff line numberDiff line change
@@ -84,11 +84,11 @@ def get(self) -> np.ndarray:
8484
return self._stack
8585

8686

87-
_PathType = Union[
87+
_PathType = Union[ # noqa: UP007
8888
# for actual paths and shapes transformed into paths
8989
svgelements.Path,
9090
# for the special case of Polygon and Polylines
91-
list[Union[svgelements.PathSegment, svgelements.Polygon, svgelements.Polyline]],
91+
list[svgelements.PathSegment | svgelements.Polygon | svgelements.Polyline],
9292
]
9393
_PathListType = list[_PathType]
9494

@@ -178,7 +178,7 @@ def _element_to_paths(elem: svgelements.SVGElement) -> _PathType | None:
178178
if isinstance(elem, svgelements.Path):
179179
if len(elem) != 0:
180180
return elem
181-
elif isinstance(elem, (svgelements.Polyline, svgelements.Polygon)):
181+
elif isinstance(elem, svgelements.Polyline | svgelements.Polygon):
182182
# Here we add a "fake" path containing just the Polyline/Polygon,
183183
# to be treated specifically by _convert_flattened_paths.
184184
path = [svgelements.Move(elem.points[0]), elem]
@@ -316,14 +316,14 @@ def _process_path(path):
316316
point_stack = _ComplexStack()
317317

318318
point_stack.append(complex(seg.end))
319-
elif isinstance(seg, (svgelements.Line, svgelements.Close)):
319+
elif isinstance(seg, svgelements.Line | svgelements.Close):
320320
start = complex(seg.start)
321321
end = complex(seg.end)
322322
if not point_stack.ends_with(start):
323323
point_stack.append(start)
324324
if end != start:
325325
point_stack.append(end)
326-
elif isinstance(seg, (svgelements.Polygon, svgelements.Polyline)):
326+
elif isinstance(seg, svgelements.Polygon | svgelements.Polyline):
327327
line = np.array(seg.points, dtype=float)
328328
line = line.view(dtype=complex).reshape(len(line))
329329
if point_stack.ends_with(line[0]):

‎vpype/metadata.py

+1-1
Original file line numberDiff line numberDiff line change
@@ -39,7 +39,7 @@ def __init__(
3939
alpha: int | None = None,
4040
):
4141
svgc = None
42-
if isinstance(red, (svgelements.Color, Color)):
42+
if isinstance(red, svgelements.Color | Color):
4343
svgc = red
4444
elif isinstance(red, str):
4545
svgc = svgelements.Color(red)

‎vpype/model.py

+6-6
Original file line numberDiff line numberDiff line change
@@ -4,8 +4,8 @@
44

55
import math
66
import pathlib
7-
from collections.abc import Iterable, Iterator
8-
from typing import Any, Callable, Optional, Union, cast
7+
from collections.abc import Callable, Iterable, Iterator
8+
from typing import Any, Union, cast
99

1010
import numpy as np
1111
from shapely.geometry import LinearRing, LineString, MultiLineString
@@ -28,7 +28,7 @@
2828
"_MetadataMixin", # for documentation
2929
]
3030

31-
LineLike = Union[LineString, LinearRing, Iterable[complex]]
31+
LineLike = Union[LineString, LinearRing, Iterable[complex]] # noqa: UP007
3232

3333
# We accept LineString and LinearRing as line collection because MultiLineString are regularly
3434
# converted to LineString/LinearRing when operation reduce them to single-line construct.
@@ -193,7 +193,7 @@ def append(self, line: LineLike) -> None:
193193
Args:
194194
line (LineLike): line to append
195195
"""
196-
if isinstance(line, (LineString, LinearRing)):
196+
if isinstance(line, LineString | LinearRing):
197197
# noinspection PyTypeChecker
198198
self._lines.append(np.array(line.coords).view(dtype=complex).reshape(-1))
199199
else:
@@ -222,7 +222,7 @@ def extend(self, lines: LineCollectionLike) -> None:
222222
# handle shapely objects
223223
if isinstance(lines, MultiLineString):
224224
lines = lines.geoms
225-
elif isinstance(lines, (LineString, LinearRing)):
225+
elif isinstance(lines, LineString | LinearRing):
226226
lines = [lines]
227227

228228
for line in lines:
@@ -623,7 +623,7 @@ def page_size(self) -> tuple[float, float] | None:
623623
return self.property(METADATA_FIELD_PAGE_SIZE)
624624

625625
@page_size.setter
626-
def page_size(self, page_size=Optional[tuple[float, float]]) -> None:
626+
def page_size(self, page_size=tuple[float, float] | None) -> None:
627627
"""Sets the page size to a new value."""
628628
self.set_property(METADATA_FIELD_PAGE_SIZE, page_size)
629629

‎vpype/text.py

+1-1
Original file line numberDiff line numberDiff line change
@@ -23,9 +23,9 @@
2323

2424
import itertools
2525
import pickle
26+
from collections.abc import Callable
2627
from dataclasses import dataclass
2728
from pathlib import Path
28-
from typing import Callable
2929

3030
import pyphen
3131

‎vpype/utils.py

+2-2
Original file line numberDiff line numberDiff line change
@@ -2,7 +2,7 @@
22

33
import math
44
import re
5-
from typing import Callable
5+
from collections.abc import Callable
66

77
import numpy as np
88

@@ -74,7 +74,7 @@ def _mm_to_px(x: float, y: float) -> tuple[float, float]:
7474

7575
def _convert_unit(value: str | float | int, units: dict[str, float]) -> float:
7676
"""Converts a string with unit to a value"""
77-
if isinstance(value, (float, int)):
77+
if isinstance(value, float | int):
7878
return value
7979

8080
mo = _FLOAT_WITH_UNIT_RE.match(value.strip().lower())

‎vpype_cli/cli.py

+3-7
Original file line numberDiff line numberDiff line change
@@ -8,8 +8,8 @@
88
import shlex
99
import sys
1010
import traceback
11-
from collections.abc import Iterable
12-
from typing import TYPE_CHECKING, Any, Callable, TextIO, Union, cast
11+
from collections.abc import Callable, Iterable
12+
from typing import TYPE_CHECKING, Any, TextIO, Union, cast
1313

1414
import click
1515
import numpy as np
@@ -204,11 +204,7 @@ def cli(
204204

205205
# since python 3.10, a new "selectable" API is used for entry points, see:
206206
# https://docs.python.org/3/library/importlib.metadata.html#entry-points
207-
# Not using it yields a deprecation warning in some circumstances.
208-
if sys.version_info >= (3, 10):
209-
entry_points = importlib.metadata.entry_points().select(group="vpype.plugins")
210-
else: # pragma: no cover
211-
entry_points = importlib.metadata.entry_points().get("vpype.plugins", [])
207+
entry_points = importlib.metadata.entry_points().select(group="vpype.plugins")
212208
for entry_point in entry_points:
213209
# noinspection PyBroadException
214210
try:

‎vpype_cli/metadata.py

+2-1
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,8 @@
11
from __future__ import annotations
22

33
import logging
4-
from typing import Any, Callable
4+
from collections.abc import Callable
5+
from typing import Any
56

67
import click
78

‎vpype_cli/substitution.py

+2-2
Original file line numberDiff line numberDiff line change
@@ -4,8 +4,8 @@
44
import os
55
import pathlib
66
import sys
7-
from collections.abc import Iterable
8-
from typing import TYPE_CHECKING, Any, Callable
7+
from collections.abc import Callable, Iterable
8+
from typing import TYPE_CHECKING, Any
99

1010
import asteval
1111

‎vpype_viewer/_painters.py

+2-2
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,7 @@
11
from __future__ import annotations
22

33
import math
4-
from typing import TYPE_CHECKING, Any, Union, cast
4+
from typing import TYPE_CHECKING, Any, cast
55

66
import moderngl as mgl
77
import numpy as np
@@ -14,7 +14,7 @@
1414
if TYPE_CHECKING: # pragma: no cover
1515
from .engine import Engine
1616

17-
ResourceType = Union[mgl.Buffer, mgl.Texture, mgl.TextureArray]
17+
ResourceType = mgl.Buffer | mgl.Texture | mgl.TextureArray
1818

1919

2020
class Painter:

‎vpype_viewer/engine.py

+1-1
Original file line numberDiff line numberDiff line change
@@ -4,7 +4,7 @@
44

55
import enum
66
from collections import defaultdict
7-
from typing import Callable
7+
from collections.abc import Callable
88

99
import moderngl as mgl
1010
import numpy as np

‎vpype_viewer/qtviewer/utils.py

+1-1
Original file line numberDiff line numberDiff line change
@@ -3,8 +3,8 @@
33
import os
44
import signal
55
import socket
6+
from collections.abc import Callable
67
from contextlib import contextmanager
7-
from typing import Callable
88

99
from PySide6 import QtNetwork
1010
from PySide6.QtGui import QAction, QActionGroup, QGuiApplication, QIcon, QPalette

0 commit comments

Comments
 (0)
Please sign in to comment.