Merge pull request #11 from Eledio/feature/tests

Unit tests for API
This commit is contained in:
Richard Kubíček
2025-03-13 11:34:02 +01:00
committed by GitHub
6 changed files with 124 additions and 2 deletions

View File

@@ -20,7 +20,7 @@ jobs:
pip install pylint
- name: pylint
run: |
pylint $(git ls-files '*.py') --fail-under=9.0 --output-format=json > pylint-report.json
pylint $(git ls-files '*.py' | grep -v '^test/') --fail-under=9.0 --output-format=json > pylint-report.json
- name: Generate Pylint Score
id: pylint_score

29
.github/workflows/pytest.yml vendored Normal file
View File

@@ -0,0 +1,29 @@
name: PyTest
on: [push]
jobs:
test:
runs-on: ubuntu-latest
steps:
- uses: actions/checkout@v4
- name: Set up Python
uses: actions/setup-python@v5
with:
python-version: '3.10'
- name: Install dependencies
run: |
python -m pip install --upgrade pip
pip install -r requirements_dev.txt
- name: Test with pytest
run: |
mkdir -p junit
pytest test/ --doctest-modules --junitxml=junit/test-results.xml --cov=superfaktura --cov-report=xml --cov-report=html
- name: Upload pytest test results
uses: actions/upload-artifact@v4
with:
name: pytest-results
path: junit/test-results.xml
# Use always() to always run this step to publish test results when there are test failures
if: ${{ always() }}

View File

@@ -1,2 +1,3 @@
requests~=2.32.3
python-dotenv~=1.0.1
python-dotenv~=1.0.1
setuptools~=75.8.0

4
requirements_dev.txt Normal file
View File

@@ -0,0 +1,4 @@
requests~=2.32.3
python-dotenv~=1.0.1
pytest~=8.3.5
pytest-cov

0
test/__init__.py Normal file
View File

View File

@@ -0,0 +1,88 @@
import os
import pytest
from unittest.mock import patch, mock_open, MagicMock
import requests
from superfaktura.superfaktura_api import (
SuperFakturaAPI,
SuperFakturaAPIException,
SuperFakturaAPIMissingCredentialsException,
)
@pytest.fixture
def api():
with patch.dict(os.environ, {
"SUPERFAKTURA_API_KEY": "test_key",
"SUPERFAKTURA_API_URL": "https://api.superfaktura.cz",
"SUPERFAKTURA_API_EMAIL": "test_email",
"SUPERFAKTURA_API_COMPANY_ID": "test_company_id"
}):
return SuperFakturaAPI()
def test_missing_credentials():
with patch.dict(os.environ, {}, clear=True):
with pytest.raises(SuperFakturaAPIMissingCredentialsException):
SuperFakturaAPI()
def test_get(api):
with patch("requests.get") as mock_get:
mock_get.return_value.status_code = 200
mock_get.return_value.json.return_value = {"data": "test"}
response = api.get("test_endpoint")
assert response == {"data": "test"}
def test_get_failure(api):
with patch("requests.get") as mock_get:
mock_get.return_value.status_code = 404
with pytest.raises(SuperFakturaAPIException):
api.get("test_endpoint")
def test_download(api):
with patch("requests.get") as mock_get:
mock_get.return_value.status_code = 200
mock_get.return_value.content = b"test_content"
with patch("builtins.open", mock_open()) as mock_file:
with open("test_file", "wb") as f:
api.download("test_endpoint", f)
mock_file().write.assert_called_once_with(b"test_content")
def test_download_failure(api):
with patch("requests.get") as mock_get:
mock_get.return_value.status_code = 404
with patch("builtins.open", mock_open()):
with open("test_file", "wb") as f:
with pytest.raises(SuperFakturaAPIException):
api.download("test_endpoint", f)
def test_post(api):
with patch("requests.post") as mock_post:
mock_post.return_value.status_code = 200
mock_post.return_value.json.return_value = {"data": "test"}
response = api.post("test_endpoint", '{"name": "Example"}')
assert response == {"data": "test"}
def test_post_failure(api):
with patch("requests.post") as mock_post:
mock_post.return_value.status_code = 404
mock_post.return_value.json.return_value = {"error": "not found"}
with pytest.raises(SuperFakturaAPIException):
api.post("test_endpoint", '{"name": "Example"}')
def test_get_invalid_json(api):
with patch("requests.get") as mock_get:
mock_response = MagicMock()
mock_response.status_code = 200
mock_response.json.side_effect = requests.exceptions.JSONDecodeError("msg", "doc", 0)
mock_get.return_value = mock_response
with pytest.raises(SuperFakturaAPIException, match="Unable to decode response as JSON"):
api.get("test_endpoint")
def test_download_not_writable_descriptor(api):
with patch("requests.get") as mock_get:
mock_get.return_value.status_code = 200
mock_get.return_value.content = b"test_content"
mock_descriptor = MagicMock()
mock_descriptor.writable.return_value = False
with pytest.raises(SuperFakturaAPIException, match=" is not writable"):
api.download("test_endpoint", mock_descriptor)