From 490f92935e9432af27ca3fe564b5168ab8dd4339 Mon Sep 17 00:00:00 2001 From: dev Date: Sat, 25 Apr 2026 23:41:02 +0200 Subject: [PATCH] feat: add uv-based project with pytest and CI workflow - Add pyproject.toml and uv.lock for Python project with uv - Add src/__init__.py with add(), multiply(), greet() functions - Add tests/test_example.py with unit tests - Add commit-message skill for Conventional Commits - Add Gitea Actions workflow to run pytest --- .gitea/workflows/test.yaml | 18 +++++-- .opencode/skills/commit-message.md | 59 ++++++++++++++++++++++ pyproject.toml | 14 ++++++ src/__init__.py | 13 +++++ tests/test_example.py | 17 +++++++ uv.lock | 78 ++++++++++++++++++++++++++++++ 6 files changed, 194 insertions(+), 5 deletions(-) create mode 100644 .opencode/skills/commit-message.md create mode 100644 pyproject.toml create mode 100644 src/__init__.py create mode 100644 tests/test_example.py create mode 100644 uv.lock diff --git a/.gitea/workflows/test.yaml b/.gitea/workflows/test.yaml index c8fe950..66bf322 100644 --- a/.gitea/workflows/test.yaml +++ b/.gitea/workflows/test.yaml @@ -9,8 +9,16 @@ jobs: - name: Checkout code uses: actions/checkout@v4 - - name: Run test - run: | - echo "Hello from Gitea Actions!" - echo "Testing runner..." - echo "Done!" \ No newline at end of file + - name: Setup Python + uses: actions/setup-python@v5 + with: + python-version: '3.11' + + - name: Install uv + uses: astral-sh/setup-uv@v4 + + - name: Install dependencies + run: uv sync --extra test + + - name: Run tests + run: uv run pytest tests/ -v \ No newline at end of file diff --git a/.opencode/skills/commit-message.md b/.opencode/skills/commit-message.md new file mode 100644 index 0000000..91ef296 --- /dev/null +++ b/.opencode/skills/commit-message.md @@ -0,0 +1,59 @@ +# skill: commit-message + +Follow these commit message conventions for all git commits: + +## Format + +``` +(): + +[optional body] + +[optional footer(s)] +``` + +## Types + +| Type | Description | +|------|------------| +| `feat` | New feature | +| `fix` | Bug fix | +| `docs` | Documentation only | +| `style` | Code style (formatting, semicolons) | +| `refactor` | Code change that neither fixes nor adds | +| `test` | Adding or updating tests | +| `chore` | Maintenance, deps, build changes | + +## Rules + +1. Use imperative: "add" not "added" or "adds" +2. Limit subject line to 50 chars +3. Capitalize first letter of description +4. No period at end of subject +5. Use scope (module/file) when applicable + +## Breaking Changes + +If change breaks backward compatibility: +- Add `BREAKING CHANGE:` in footer +- Use `!` after type: `feat!:` + +## Examples + +``` +feat(auth): add login endpoint + +fix(api): resolve null pointer in user response + +docs(readme): update installation steps + +feat!: migrate to v2 API + +BREAKING CHANGE: auth header format changed +``` + +## When to Write Body + +- Explanation of "what" and "why" (not "how") +- More detail than fits in subject line +- References to issues/PRs \ No newline at end of file diff --git a/pyproject.toml b/pyproject.toml new file mode 100644 index 0000000..d1539fb --- /dev/null +++ b/pyproject.toml @@ -0,0 +1,14 @@ +[project] +name = "explore-agentic" +version = "0.1.0" +description = "A sample Python project" +requires-python = ">=3.11" +dependencies = [] + +[project.optional-dependencies] +test = ["pytest>=8.0.0"] + +[tool.pytest.ini_options] +testpaths = ["tests"] +python_files = ["test_*.py"] +python_functions = ["test_*"] \ No newline at end of file diff --git a/src/__init__.py b/src/__init__.py new file mode 100644 index 0000000..3e7cef6 --- /dev/null +++ b/src/__init__.py @@ -0,0 +1,13 @@ +def add(a: int, b: int) -> int: + """Add two numbers together.""" + return a + b + + +def multiply(a: int, b: int) -> int: + """Multiply two numbers together.""" + return a * b + + +def greet(name: str) -> str: + """Return a greeting message.""" + return f"Hello, {name}!" \ No newline at end of file diff --git a/tests/test_example.py b/tests/test_example.py new file mode 100644 index 0000000..aeaabf4 --- /dev/null +++ b/tests/test_example.py @@ -0,0 +1,17 @@ +import pytest +from src import add, multiply, greet + + +def test_add(): + assert add(1, 2) == 3 + assert add(-1, 1) == 0 + + +def test_multiply(): + assert multiply(2, 3) == 6 + assert multiply(0, 5) == 0 + + +def test_greet(): + assert greet("World") == "Hello, World!" + assert greet("") == "Hello, !" \ No newline at end of file diff --git a/uv.lock b/uv.lock new file mode 100644 index 0000000..a629489 --- /dev/null +++ b/uv.lock @@ -0,0 +1,78 @@ +version = 1 +revision = 3 +requires-python = ">=3.11" + +[[package]] +name = "colorama" +version = "0.4.6" +source = { registry = "https://pypi.org/simple" } +sdist = { url = "https://files.pythonhosted.org/packages/d8/53/6f443c9a4a8358a93a6792e2acffb9d9d5cb0a5cfd8802644b7b1c9a02e4/colorama-0.4.6.tar.gz", hash = "sha256:08695f5cb7ed6e0531a20572697297273c47b8cae5a63ffc6d6ed5c201be6e44", size = 27697, upload-time = "2022-10-25T02:36:22.414Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/d1/d6/3965ed04c63042e047cb6a3e6ed1a63a35087b6a609aa3a15ed8ac56c221/colorama-0.4.6-py2.py3-none-any.whl", hash = "sha256:4f1d9991f5acc0ca119f9d443620b77f9d6b33703e51011c16baf57afb285fc6", size = 25335, upload-time = "2022-10-25T02:36:20.889Z" }, +] + +[[package]] +name = "explore-agentic" +version = "0.1.0" +source = { virtual = "." } + +[package.optional-dependencies] +test = [ + { name = "pytest" }, +] + +[package.metadata] +requires-dist = [{ name = "pytest", marker = "extra == 'test'", specifier = ">=8.0.0" }] +provides-extras = ["test"] + +[[package]] +name = "iniconfig" +version = "2.3.0" +source = { registry = "https://pypi.org/simple" } +sdist = { url = "https://files.pythonhosted.org/packages/72/34/14ca021ce8e5dfedc35312d08ba8bf51fdd999c576889fc2c24cb97f4f10/iniconfig-2.3.0.tar.gz", hash = "sha256:c76315c77db068650d49c5b56314774a7804df16fee4402c1f19d6d15d8c4730", size = 20503, upload-time = "2025-10-18T21:55:43.219Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/cb/b1/3846dd7f199d53cb17f49cba7e651e9ce294d8497c8c150530ed11865bb8/iniconfig-2.3.0-py3-none-any.whl", hash = "sha256:f631c04d2c48c52b84d0d0549c99ff3859c98df65b3101406327ecc7d53fbf12", size = 7484, upload-time = "2025-10-18T21:55:41.639Z" }, +] + +[[package]] +name = "packaging" +version = "26.2" +source = { registry = "https://pypi.org/simple" } +sdist = { url = "https://files.pythonhosted.org/packages/d7/f1/e7a6dd94a8d4a5626c03e4e99c87f241ba9e350cd9e6d75123f992427270/packaging-26.2.tar.gz", hash = "sha256:ff452ff5a3e828ce110190feff1178bb1f2ea2281fa2075aadb987c2fb221661", size = 228134, upload-time = "2026-04-24T20:15:23.917Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/df/b2/87e62e8c3e2f4b32e5fe99e0b86d576da1312593b39f47d8ceef365e95ed/packaging-26.2-py3-none-any.whl", hash = "sha256:5fc45236b9446107ff2415ce77c807cee2862cb6fac22b8a73826d0693b0980e", size = 100195, upload-time = "2026-04-24T20:15:22.081Z" }, +] + +[[package]] +name = "pluggy" +version = "1.6.0" +source = { registry = "https://pypi.org/simple" } +sdist = { url = "https://files.pythonhosted.org/packages/f9/e2/3e91f31a7d2b083fe6ef3fa267035b518369d9511ffab804f839851d2779/pluggy-1.6.0.tar.gz", hash = "sha256:7dcc130b76258d33b90f61b658791dede3486c3e6bfb003ee5c9bfb396dd22f3", size = 69412, upload-time = "2025-05-15T12:30:07.975Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/54/20/4d324d65cc6d9205fabedc306948156824eb9f0ee1633355a8f7ec5c66bf/pluggy-1.6.0-py3-none-any.whl", hash = "sha256:e920276dd6813095e9377c0bc5566d94c932c33b27a3e3945d8389c374dd4746", size = 20538, upload-time = "2025-05-15T12:30:06.134Z" }, +] + +[[package]] +name = "pygments" +version = "2.20.0" +source = { registry = "https://pypi.org/simple" } +sdist = { url = "https://files.pythonhosted.org/packages/c3/b2/bc9c9196916376152d655522fdcebac55e66de6603a76a02bca1b6414f6c/pygments-2.20.0.tar.gz", hash = "sha256:6757cd03768053ff99f3039c1a36d6c0aa0b263438fcab17520b30a303a82b5f", size = 4955991, upload-time = "2026-03-29T13:29:33.898Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/f4/7e/a72dd26f3b0f4f2bf1dd8923c85f7ceb43172af56d63c7383eb62b332364/pygments-2.20.0-py3-none-any.whl", hash = "sha256:81a9e26dd42fd28a23a2d169d86d7ac03b46e2f8b59ed4698fb4785f946d0176", size = 1231151, upload-time = "2026-03-29T13:29:30.038Z" }, +] + +[[package]] +name = "pytest" +version = "9.0.3" +source = { registry = "https://pypi.org/simple" } +dependencies = [ + { name = "colorama", marker = "sys_platform == 'win32'" }, + { name = "iniconfig" }, + { name = "packaging" }, + { name = "pluggy" }, + { name = "pygments" }, +] +sdist = { url = "https://files.pythonhosted.org/packages/7d/0d/549bd94f1a0a402dc8cf64563a117c0f3765662e2e668477624baeec44d5/pytest-9.0.3.tar.gz", hash = "sha256:b86ada508af81d19edeb213c681b1d48246c1a91d304c6c81a427674c17eb91c", size = 1572165, upload-time = "2026-04-07T17:16:18.027Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/d4/24/a372aaf5c9b7208e7112038812994107bc65a84cd00e0354a88c2c77a617/pytest-9.0.3-py3-none-any.whl", hash = "sha256:2c5efc453d45394fdd706ade797c0a81091eccd1d6e4bccfcd476e2b8e0ab5d9", size = 375249, upload-time = "2026-04-07T17:16:16.13Z" }, +]