16 Commits
v1.0.0 ... main

Author SHA1 Message Date
Richard Kubíček
adc2fa4aac Merge pull request #12 from Eledio/test/clientContact
Unit tests for Client Contact
2025-03-13 22:20:45 +01:00
Richard Kubíček
3d4400db2d Merge pull request #11 from Eledio/feature/tests
Unit tests for API
2025-03-13 11:34:02 +01:00
Richard Kubíček
f40b015beb test_client_contact: add docstring 2025-03-12 22:03:25 +01:00
Richard Kubíček
964882bc5d test_client_contact: fix pytest.fixture 2025-03-12 22:02:29 +01:00
Richard Kubíček
c861f35ad2 test_client_contact: add api credinitials 2025-03-12 21:51:55 +01:00
Richard Kubíček
133702dc1d test_superfaktura_api: fix format 2025-03-12 21:49:10 +01:00
Richard Kubíček
765334ed92 test_client_contact: add tests 2025-03-12 21:49:00 +01:00
Richard Kubíček
b41bf28a34 test_superfaktura_api: fix unnecessary mock file open 2025-03-11 21:23:56 +01:00
Richard Kubíček
8d3b2589b0 test_superfaktura_api: add test download with not writable file descriptor 2025-03-11 21:18:14 +01:00
Richard Kubíček
d2cbbd0596 test_superfaktura_api: add test invalid json as repsonse from the server 2025-03-11 21:18:14 +01:00
Richard Kubíček
45259f8f27 Update pytest.yml 2025-03-05 22:24:27 +01:00
Richard Kubíček
5b50f2c3eb ga: fix workflow for pylint 2025-03-05 13:15:43 +01:00
Richard Kubíček
16cb1b39ba ga: add workflow for pytest 2025-03-05 13:07:12 +01:00
Richard Kubíček
e2f89cf754 requirements_dev: add pytest-cov 2025-03-05 13:00:04 +01:00
Richard Kubíček
886004a20a test: introduce basic tests 2025-03-05 12:07:46 +01:00
Richard Kubíček
e668f76b4c requirements: add setuptools 2025-03-05 12:07:27 +01:00
7 changed files with 210 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,69 @@
import os
import pytest # type: ignore
from unittest.mock import patch
from superfaktura.client_contacts import (
ClientContactModel,
ClientContact,
ClientException,
)
"""
Tests for the ClientContact class from the superfaktura package.
These tests validate the functionality of methods for managing client contacts
through the SuperFaktura API.
"""
@pytest.fixture
def client_contact():
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 ClientContact()
def test_add_contact_success(client_contact):
client = ClientContactModel(
name="John Doe",
)
with patch("superfaktura.superfaktura_api.SuperFakturaAPI.post") as mock_post:
mock_post.return_value = {"error_message": "Client created"}
assert client_contact.add_contact(contact=client)
def test_add_contact_failed(client_contact):
client = ClientContactModel(
name="John Doe",
)
with patch("superfaktura.superfaktura_api.SuperFakturaAPI.post") as mock_post:
mock_post.return_value = {"error_message": "Client creation failed"}
assert not client_contact.add_contact(contact=client)
def test_list(client_contact):
with patch("superfaktura.superfaktura_api.SuperFakturaAPI.get") as mock_get:
mock_get.return_value = {"data": "test"}
assert client_contact.list() == {"data": "test"}
def test_get_client_exists(client_contact):
with patch("superfaktura.superfaktura_api.SuperFakturaAPI.get") as mock_get:
mock_get.return_value = {"Client": {"name": "John Doe", "id": 1}}
assert client_contact.get_client(client_id=1).name == "John Doe"
def test_get_client_not_exists(client_contact):
with patch("superfaktura.superfaktura_api.SuperFakturaAPI.get") as mock_get:
mock_get.return_value = {}
with pytest.raises(ClientException):
client_contact.get_client(client_id=1)

View File

@@ -0,0 +1,105 @@
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)