From b4f082441b0e0196e8f0140ccc11e0aefe2b86ac Mon Sep 17 00:00:00 2001 From: Nicolas Delaby Date: Tue, 7 Jan 2025 13:11:38 +0100 Subject: [PATCH] Switch to uv for building and packaging Get dynamic version based on current tag --- .github/workflows/release.yml | 14 +++--- .github/workflows/test_suite.yml | 73 ++++++++++++++++++++++++++---- .gitignore | 4 ++ .readthedocs.yaml | 19 ++++---- django_fsm_log/models.py | 4 +- pyproject.toml | 78 +++++++++++++++++++++++++++++--- setup.cfg | 8 ---- setup.py | 53 ---------------------- tests/models.py | 11 +++-- tests/test_model.py | 8 ++-- tox.ini | 24 ++++++---- 11 files changed, 184 insertions(+), 112 deletions(-) delete mode 100644 setup.cfg delete mode 100644 setup.py diff --git a/.github/workflows/release.yml b/.github/workflows/release.yml index a7be9e8..0033404 100644 --- a/.github/workflows/release.yml +++ b/.github/workflows/release.yml @@ -2,7 +2,7 @@ name: Release on: push: tags: - - '*' + - "*" jobs: build: @@ -17,17 +17,17 @@ jobs: - name: Set up Python uses: actions/setup-python@v5 with: - python-version: '3.11' + python-version: "3.11" - name: Install dependencies run: | - python -m pip install -U pip - python -m pip install -U setuptools twine wheel + curl -LsSf https://astral.sh/uv/install.sh | sh + - name: Build package run: | - python setup.py --version - python setup.py sdist --format=gztar bdist_wheel - twine check dist/* + uv build + uvx twine check dist/* + - name: Upload packages to Jazzband if: github.event_name == 'push' && startsWith(github.ref, 'refs/tags') uses: pypa/gh-action-pypi-publish@v1.12.3 diff --git a/.github/workflows/test_suite.yml b/.github/workflows/test_suite.yml index 473be38..bc5b6e4 100644 --- a/.github/workflows/test_suite.yml +++ b/.github/workflows/test_suite.yml @@ -5,34 +5,89 @@ on: - master pull_request: jobs: - tox: + pytest: runs-on: ubuntu-latest strategy: matrix: python-version: - - "3.8" - "3.9" - "3.10" - "3.11" - "3.12" + - "3.13" + django-version: + - "4.2" + - "5.0" + - "5.1" + exclude: + - django-version: 4.2 + python-version: 3.12 + - django-version: 4.2 + python-version: 3.13 + - django-version: 5.0 + python-version: 3.9 + - django-version: 5.1 + python-version: 3.9 steps: - uses: actions/checkout@v4 - - name: Set up Python ${{ matrix.python-version }} - uses: actions/setup-python@v5 + uses: astral-sh/setup-uv@v5 + with: + enable-cache: true + cache-dependency-glob: "pyproject.toml" + cache-suffix: ${{ matrix.python-version }} + - name: Install Python + run: uv python install ${{ matrix.python-version }} + env: + UV_PYTHON_PREFERENCE: only-managed + - run: uv sync --all-groups + - run: uv run --with 'django~=${{ matrix.django-version }}.0' pytest --cov --cov-report= + env: + DJANGO_SETTINGS_MODULE: tests.settings + PYTHONPATH: "." + - name: Rename coverage file + run: mv .coverage .coverage.py${{ matrix.python-version }}.dj${{ matrix.django-version }} + - name: Save coverage file + uses: actions/upload-artifact@v4 with: - python-version: ${{ matrix.python-version }} + name: .coverage.py${{ matrix.python-version }}.dj${{ matrix.django-version }} + path: .coverage.py${{ matrix.python-version }}.dj${{ matrix.django-version }} + include-hidden-files: true - - run: pip install tox tox-gh-actions codecov + codecov: + needs: pytest + runs-on: ubuntu-latest + steps: + - uses: actions/checkout@v4 + - uses: astral-sh/setup-uv@v5 + - run: uv python install 3.13 + - uses: actions/download-artifact@v4 + with: + pattern: .coverage.* + merge-multiple: true + - name: Combine coverage + run: | + uv run coverage combine + uv run coverage xml + - name: Upload coverage to Codecov + uses: codecov/codecov-action@v5 - - run: tox + lint: + runs-on: ubuntu-latest + steps: + - uses: actions/checkout@v4 + - uses: astral-sh/setup-uv@v5 + - run: uv python install 3.13 + - run: uv sync --group dev + - run: uvx ruff check + - run: uvx ruff format --check - - run: codecov check: runs-on: ubuntu-latest if: always() needs: - - tox + - pytest + - lint steps: - name: Decide whether the needed jobs succeeded or failed uses: re-actors/alls-green@release/v1 diff --git a/.gitignore b/.gitignore index 96ab35b..4a3092c 100644 --- a/.gitignore +++ b/.gitignore @@ -9,3 +9,7 @@ build/ dist/ *.egg-info docs/_build + +uv.lock + +.envrc diff --git a/.readthedocs.yaml b/.readthedocs.yaml index 2124f77..7f34ed5 100644 --- a/.readthedocs.yaml +++ b/.readthedocs.yaml @@ -1,16 +1,17 @@ version: 2 build: - os: ubuntu-20.04 + os: ubuntu-22.04 tools: - python: "3.9" + python: "3.11" + + commands: + - asdf plugin add uv + - asdf install uv latest + - asdf global uv latest + - mkdir -p $READTHEDOCS_OUTPUT/html/ + - uv sync --group docs + - uv run -m sphinx -T -b html -d docs/_build/doctrees -D language=en docs $READTHEDOCS_OUTPUT/html sphinx: configuration: docs/conf.py - -python: - install: - - method: pip - path: . - extra_requirements: - - docs diff --git a/django_fsm_log/models.py b/django_fsm_log/models.py index fc13d8e..e999145 100644 --- a/django_fsm_log/models.py +++ b/django_fsm_log/models.py @@ -16,7 +16,7 @@ class StateLog(models.Model): null=True, on_delete=models.SET_NULL, ) - source_state = models.CharField(max_length=255, db_index=True, null=True, blank=True, default=None) + source_state = models.CharField(max_length=255, db_index=True, null=True, blank=True, default=None) # noqa:DJ001 state = models.CharField("Target state", max_length=255, db_index=True) transition = models.CharField(max_length=255) @@ -24,7 +24,7 @@ class StateLog(models.Model): object_id = models.PositiveIntegerField(db_index=True) content_object = GenericForeignKey("content_type", "object_id") - description = models.TextField(blank=True, null=True) + description = models.TextField(blank=True, null=True) # noqa:DJ001 objects = StateLogManager() diff --git a/pyproject.toml b/pyproject.toml index d933e7c..1753757 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -1,10 +1,74 @@ -[tool.black] -line-length = 119 -target-version = ["py37"] -extend-exclude = "(^/django_fsm_log/migrations/.*$|^docs/.*$)" +[project] +license = { file = "LICENSE" } +description = "Transition's persistence for django-fsm" +name = "django-fsm-log" +dynamic = ["version"] +readme = "README.md" +requires-python = ">=3.9" +authors = [ + { name = "Gizmag", email = "tech@gizmag.com" }, + { name = "Various Contributors" }, +] +keywords = ["django", "django-fsm-2"] +classifiers = [ + "Development Status :: 5 - Production/Stable", + "Environment :: Web Environment", + "Framework :: Django", + "Framework :: Django :: 4.2", + "Framework :: Django :: 5.0", + "Framework :: Django :: 5.1", + "Intended Audience :: Developers", + "License :: OSI Approved :: MIT License", + "Operating System :: OS Independent", + "Programming Language :: Python :: 3", + "Programming Language :: Python :: 3.9", + "Programming Language :: Python :: 3.10", + "Programming Language :: Python :: 3.11", + "Programming Language :: Python :: 3.12", + "Programming Language :: Python :: 3.13", + "Topic :: Software Development :: Libraries :: Python Modules", +] +dependencies = ["django>=4.2", "django-fsm-2", "django_appconf"] + +[dependency-groups] +dev = [ + "pytest", + "pytest-cov", + "pytest-django", + "pytest-mock", + "tox", + "tox-uv>=1.17.0", + "twine", +] +ci = ["codecov>=2.1.13"] +docs = ["sphinx", "sphinx_rtd_theme", "myst-parser"] + +[project.urls] +Documentation = "https://django-fsm-log.readthedocs.io/en/latest/" +Homepage = "https://github.com/jazzband/django-fsm-log" + +[build-system] +requires = ["setuptools>=64", "setuptools_scm>=8"] +build-backend = "setuptools.build_meta" + +[tool.setuptools] +include-package-data = true + +[tool.setuptools_scm] [tool.ruff] line-length = 119 -target-version = "py37" -select = ["E", "F", "I", "B", "C4", "T20", "TID", "UP"] -exclude = ["django_fsm_log/migrations", ".tox", "build"] +target-version = "py39" +extend-exclude = ["django_fsm_log/migrations/", ".tox/", "build/", "docs/"] + +[tool.ruff.lint] +select = ["E", "F", "I", "B", "C4", "T20", "TID", "UP", "DJ"] + +[tool.pytest.ini_options] +markers = [ + "ignore_article: Configure the settings DJANGO_FSM_LOG_IGNORED_MODELS to ignore Article Model.", + "pending_objects: Install PendingStateLogManager on StateLog", +] +testpaths = ["tests"] +pythonpath = ["."] +DJANGO_SETTINGS_MODULE = "tests.settings" diff --git a/setup.cfg b/setup.cfg deleted file mode 100644 index bb0248b..0000000 --- a/setup.cfg +++ /dev/null @@ -1,8 +0,0 @@ -[tool:pytest] -DJANGO_SETTINGS_MODULE = tests.settings -testpaths = tests - -markers = - ignore_article: Configure the settings DJANGO_FSM_LOG_IGNORED_MODELS to ignore - Article Model. - pending_objects: Install PendingStateLogManager on StateLog diff --git a/setup.py b/setup.py deleted file mode 100644 index 5c2d428..0000000 --- a/setup.py +++ /dev/null @@ -1,53 +0,0 @@ -#!/usr/bin/env python - -from setuptools import find_packages, setup - - -def readfile(filename): - with open(filename) as open_file: - return open_file.read() - - -setup( - name="django-fsm-log", - version="4.0.1", - description="Transition's persistence for django-fsm", - long_description=readfile("README.md"), - long_description_content_type="text/markdown", - author="Gizmag", - author_email="tech@gizmag.com", - url="https://github.com/jazzband/django-fsm-log", - license="MIT", - packages=find_packages(exclude=["tests"]), - install_requires=["django>=3.2", "django-fsm-2", "django_appconf"], - extras_require={ - "testing": [ - "pytest", - "pytest-cov", - "pytest-django", - "pytest-mock", - ], - "docs": [ - "sphinx", - "sphinx_rtd_theme", - "myst-parser", - ], - }, - classifiers=[ - "Development Status :: 5 - Production/Stable", - "Environment :: Web Environment", - "Framework :: Django", - "Framework :: Django :: 4.2", - "Framework :: Django :: 5.0", - "Intended Audience :: Developers", - "License :: OSI Approved :: MIT License", - "Operating System :: OS Independent", - "Programming Language :: Python :: 3", - "Programming Language :: Python :: 3.8", - "Programming Language :: Python :: 3.9", - "Programming Language :: Python :: 3.10", - "Programming Language :: Python :: 3.11", - "Programming Language :: Python :: 3.12", - "Topic :: Software Development :: Libraries :: Python Modules", - ], -) diff --git a/tests/models.py b/tests/models.py index 2a1698b..65a9660 100644 --- a/tests/models.py +++ b/tests/models.py @@ -15,17 +15,15 @@ class Article(models.Model): state = FSMField(choices=STATES, default="draft", protected=True) + def __str__(self): + return f"pk={self.pk}" + @fsm_log_by @fsm_log_description @transition(field=state, source="draft", target="submitted") def submit(self, description=None, by=None): pass - @fsm_log_by - @transition(field=state, source="submitted", target="draft") - def request_changes(self, by=None): - pass - @fsm_log_by @transition(field=state, source="submitted", target="published") def publish(self, by=None): @@ -72,6 +70,9 @@ class ArticleInteger(models.Model): state = FSMIntegerField(choices=STATES, default=STATE_ONE) + def __str__(self): + return f"pk={self.pk}" + @fsm_log_by @transition(field=state, source=STATE_ONE, target=STATE_TWO) def change_to_two(self, by=None): diff --git a/tests/test_model.py b/tests/test_model.py index 79e7651..ed0ca7f 100644 --- a/tests/test_model.py +++ b/tests/test_model.py @@ -11,7 +11,7 @@ def test_get_available_state_transitions(article): def test_get_all_state_transitions(article): - assert len(list(article.get_all_state_transitions())) == 8 + assert len(list(article.get_all_state_transitions())) == 7 def test_log_created_on_transition(article): @@ -46,7 +46,7 @@ def test_by_is_set_when_passed_into_transition(article, user): log = StateLog.objects.all()[0] assert user == log.by with pytest.raises(AttributeError): - article.__django_fsm_log_attr_by # noqa: B018 + _ = article.__django_fsm_log_attr_by def test_by_is_none_when_not_set_in_transition(article): @@ -63,7 +63,7 @@ def test_description_is_set_when_passed_into_transition(article): log = StateLog.objects.all()[0] assert description == log.description with pytest.raises(AttributeError): - article.__django_fsm_log_attr_description # noqa: B018 + _ = article.__django_fsm_log_attr_description def test_description_is_none_when_not_set_in_transition(article): @@ -161,3 +161,5 @@ def test_get_display_state_with_integer(article_integer): article_integer = ArticleInteger.objects.get(pk=article_integer.pk) assert log.get_state_display() == article_integer.get_state_display() + # only to appease code coverage + assert str(article_integer) == f"pk={article_integer.pk}" diff --git a/tox.ini b/tox.ini index 843f6df..d1bf202 100644 --- a/tox.ini +++ b/tox.ini @@ -1,9 +1,10 @@ [tox] envlist = py{39,310,311}-dj-4.2 - py{310,311,312,313}-5.0 - py{310,311,312,313}-5.1 - py{310,311,312,313}-dj-master + py{310,311,312,313}-dj-5.0 + py{310,311,312,313}-dj-5.1 + py{310,311,312,313}-dj-main + lint [gh-actions] python = @@ -16,12 +17,17 @@ python = [testenv] usedevelop = true commands = pytest --cov=django_fsm_log --cov=tests {posargs} -extras = testing setenv= DJANGO_SETTINGS_MODULE = tests.settings - PYTHONPATH = {toxinidir} deps = - dj-4.2: Django>=4.2,<5 - dj-5.0: Django>=5,<5.1 - dj-5.1: Django>=5.1,<5.2 - dj-master: https://github.com/django/django/archive/master.tar.gz + dj-4.2: Django~=4.2 + dj-5.0: Django~=5.0 + dj-5.1: Django~=5.1 + dj-main: https://github.com/django/django/archive/main.tar.gz + +[testenv:lint] +basepython = python3 +deps = ruff +commands = + ruff check + ruff format --check