Skip to content
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
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
10 changes: 10 additions & 0 deletions .github/workflows/ci.yml
Original file line number Diff line number Diff line change
Expand Up @@ -65,6 +65,16 @@ jobs:
run: uvx uvtask static-analysis:types
if: ${{ env.PIPELINE_TESTS == 'true' }}

- name: complexity-visibility
shell: bash
run: uvx uvtask complexity:visibility
if: ${{ env.PIPELINE_TESTS == 'true' }}

- name: complexity-enforcement
shell: bash
run: uvx uvtask complexity:enforcement
if: ${{ env.PIPELINE_TESTS == 'true' }}

- name: unit-tests
shell: bash
run: uvx uvtask unit-tests
Expand Down
53 changes: 29 additions & 24 deletions pyproject.toml
Original file line number Diff line number Diff line change
Expand Up @@ -49,22 +49,35 @@ requires-python = ">=3.13"

[dependency-groups]
dev = [
"bandit>=1.9.2", # security-analysis
"pip-licenses>=5.5.0", # security-analysis
"pytest>=8.0.0", # test
"bandit>=1.9.2", # security-analysis:vulnerabilities
"pip-licenses>=5.5.0", # security-analysis:licenses
"pytest>=9.0.2", # test
"pytest-cov>=7.0.0", # test, coverage
"pytest-xdist>=3.8.0", # test
"ruff>=0.14.10", # code-formatter, static-analysis
"ty>=0.0.6", # static-analysis
"radon>=6.0.1", # complexity:visibility
"ruff>=0.14.10", # code-formatter, static-analysis:linter
"ty>=0.0.7", # static-analysis:types
"xenon>=0.9.3", # complexity:enforcement
]

[project.urls]
"documentation" = "https://aiopy.github.io/python-uvtask/"
"documentation" = "https://github.com/aiopy/python-uvtask/README.md"
"repository" = "https://github.com/aiopy/python-uvtask"

[tool.bandit]
exclude_dirs = ["tests"]
skips = ["B404", "B602"]
skips = []

[tool.coverage.paths]
source = ["uvtask"]
[tool.coverage.run]
data_file = ".venv/var/coverage/.coverage"
disable_warnings = ["no-data-collected"]
source = ["tests"]
[tool.coverage.report]
fail_under = 85
[tool.coverage.html]
directory = ".venv/var/coverage"

[tool.pip-licenses]
allow-only = "Apache;BSD;MIT;MPL"
Expand All @@ -73,12 +86,12 @@ ignore-packages = [
partial-match = true

[tool.pytest.ini_options]
cache_dir = "var/pytest"
cache_dir = ".venv/var/pytest"
addopts = "-q -n 1 -p no:warnings --no-cov-on-fail"
testpaths = ["tests"]

[tool.ruff]
cache-dir = "var/ruff"
cache-dir = ".venv/var/ruff"
line-length = 160
target-version = "py313"
respect-gitignore = true
Expand All @@ -94,8 +107,6 @@ lint.select = [
"RUF", # ruff-specific
]
lint.ignore = [
"PLC0415",
"PLR0912",
"PLR0913",
"PLR2004",
]
Expand All @@ -112,15 +123,14 @@ force-single-line = false
known-first-party = ["uvtask"]

[tool.ruff.lint.mccabe]
max-complexity = 20
max-complexity = 10

[tool.ty.environment]
python-version = "3.13"

[tool.ty.src]
exclude = [
"tests/fixtures/**",
"var",
".venv",
]

Expand All @@ -130,26 +140,21 @@ upgrade-install = { command = "uv sync --frozen --no-dev --upgrade --refresh", d
dev-install = { command = "uv sync --dev --all-extras", description = "Install all dependencies including dev and extras" }
upgrade-dev-install = { command = "uv sync --dev --all-extras --upgrade --refresh", description = "Upgrade and refresh installation of all dependencies including dev and extras" }
code-formatter = { command = "uv run ruff format uvtask tests", description = "Format code with ruff" }
"security-analysis" = { command = ["security-analysis:licenses", "security-analysis:vulnerabilities"], description = "Run all security analysis checks" }
security-analysis = { command = ["security-analysis:licenses", "security-analysis:vulnerabilities"], description = "Run all security analysis checks" }
"security-analysis:licenses" = { command = "uv run pip-licenses", description = "Check third-party dependencies licenses using pip-licenses" }
"security-analysis:vulnerabilities" = { command = "uv run bandit -r -c pyproject.toml uvtask tests", description = "Scan code for security vulnerabilities using bandit" }
"static-analysis" = { command = ["static-analysis:linter", "static-analysis:types"], description = "Run all static analysis checks" }
static-analysis = { command = ["static-analysis:linter", "static-analysis:types"], description = "Run all static analysis checks" }
"static-analysis:linter" = { command = "uv run ruff check uvtask tests", description = "Run linter checks using ruff" }
"static-analysis:types" = { command = "uv run ty check uvtask tests", description = "Run type checks using ty" }
complexity = { command = ["complexity:visibility", "complexity:enforcement"], description = "Run complexity analysis and enforcement" }
"complexity:visibility" = { command = ["uv run radon cc uvtask -a -nb", "uv run radon mi uvtask -nb"], description = "Show cyclomatic complexity and maintainability metrics using radon" }
"complexity:enforcement" = { command = "uv run xenon --max-absolute C --max-modules B --max-average A uvtask", description = "Ensure code complexity stays within allowed limits using xenon" }
test = { command = ["unit-tests", "integration-tests", "functional-tests"], description = "Run all tests with pytest" }
unit-tests = { command = "uv run pytest tests/unit", description = "Run unit tests with pytest" }
integration-tests = { command = "uv run pytest tests/integration", description = "Run integration tests with pytest" }
functional-tests = { command = "uv run pytest -n1 tests/functional", description = "Run functional tests with pytest" }
coverage = { command = "uv run pytest -n1 --cov --cov-report=html", description = "Run tests with coverage report in HTML using pytest" }
clean = { command = """python3 -c "
from glob import iglob
from shutil import rmtree

for pathname in ['./build', './*.egg-info', './dist', './var', '**/__pycache__']:
for path in iglob(pathname, recursive=True):
rmtree(path, ignore_errors=True)
"
""", description = "Clean build artifacts" }
clean = { command = """python -c "import shutil, glob; [shutil.rmtree(p, ignore_errors=True) for pattern in ['build', 'dist', '*.egg-info', '**/__pycache__'] for p in glob.glob(pattern, recursive=True)]" """, description = "Remove build artifacts and cache directories" }

[project.scripts]
uvtask = "uvtask.cli:main"
Expand Down
9 changes: 1 addition & 8 deletions tests/unit/test_cli.py
Original file line number Diff line number Diff line change
@@ -1,3 +1,4 @@
import os
from pathlib import Path
from unittest.mock import MagicMock

Expand All @@ -10,8 +11,6 @@ class TestCliApplication:
def test_run_missing_pyproject(self, temp_dir: Path) -> None:
original_cwd = Path.cwd()
try:
import os

os.chdir(temp_dir)
mock_script_loader = MagicMock()
mock_version_loader = MagicMock()
Expand Down Expand Up @@ -42,8 +41,6 @@ def test_run_missing_pyproject(self, temp_dir: Path) -> None:
def test_run_validates_reserved_commands(self, pyproject_toml: Path) -> None:
original_cwd = Path.cwd()
try:
import os

os.chdir(pyproject_toml.parent)
mock_script_loader = MagicMock()
mock_script_loader.load_scripts_with_descriptions.return_value = ({"help": "echo help"}, {})
Expand Down Expand Up @@ -75,8 +72,6 @@ def test_run_validates_reserved_commands(self, pyproject_toml: Path) -> None:
def test_run_executes_command(self, pyproject_toml: Path) -> None:
original_cwd = Path.cwd()
try:
import os

os.chdir(pyproject_toml.parent)
mock_script_loader = MagicMock()
mock_script_loader.load_scripts_with_descriptions.return_value = (
Expand Down Expand Up @@ -117,8 +112,6 @@ def test_run_executes_command(self, pyproject_toml: Path) -> None:
def test_run_handles_help_command(self, pyproject_toml: Path) -> None:
original_cwd = Path.cwd()
try:
import os

os.chdir(pyproject_toml.parent)
mock_script_loader = MagicMock()
mock_script_loader.load_scripts_with_descriptions.return_value = ({"test": "echo test"}, {})
Expand Down
4 changes: 2 additions & 2 deletions tests/unit/test_executor.py
Original file line number Diff line number Diff line change
Expand Up @@ -13,7 +13,7 @@ def test_execute_success(self, mock_run: MagicMock) -> None:
assert exit_code == 0
mock_run.assert_called_once()

@patch("uvtask.executor.subprocess.run")
@patch("uvtask.executor.run")
def test_execute_with_quiet(self, mock_run: MagicMock) -> None:
mock_run.return_value = CompletedProcess(["echo", "test"], 0)
executor = CommandExecutor()
Expand All @@ -22,7 +22,7 @@ def test_execute_with_quiet(self, mock_run: MagicMock) -> None:
call_kwargs = mock_run.call_args[1]
assert call_kwargs["stdout"] is not None or call_kwargs.get("stdout") is None

@patch("uvtask.executor.subprocess.run")
@patch("uvtask.executor.run")
def test_execute_with_double_quiet(self, mock_run: MagicMock) -> None:
mock_run.return_value = CompletedProcess(["echo", "test"], 0)
executor = CommandExecutor()
Expand Down
Loading