mirror of
https://github.com/YunoHost-Apps/pyinventory_ynh.git
synced 2024-09-03 20:16:09 +02:00
add pytests
This commit is contained in:
parent
add1026c11
commit
5c6087fb44
9 changed files with 289 additions and 0 deletions
41
.github/workflows/pytest.yml
vendored
Normal file
41
.github/workflows/pytest.yml
vendored
Normal file
|
@ -0,0 +1,41 @@
|
||||||
|
name: pytest
|
||||||
|
|
||||||
|
on:
|
||||||
|
schedule:
|
||||||
|
- cron: '0 8 * * *'
|
||||||
|
push:
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
test:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
strategy:
|
||||||
|
max-parallel: 2
|
||||||
|
matrix:
|
||||||
|
python-version: [3.9, 3.8, 3.7]
|
||||||
|
steps:
|
||||||
|
- uses: actions/checkout@v1
|
||||||
|
- name: 'Set up Python ${{ matrix.python-version }}'
|
||||||
|
uses: actions/setup-python@v1
|
||||||
|
with:
|
||||||
|
python-version: '${{ matrix.python-version }}'
|
||||||
|
|
||||||
|
- name: 'Install package'
|
||||||
|
run: |
|
||||||
|
pip3 install poetry
|
||||||
|
make install
|
||||||
|
|
||||||
|
- name: 'List installed packages'
|
||||||
|
run: |
|
||||||
|
poetry run pip freeze
|
||||||
|
|
||||||
|
- name: 'Run tests with Python v${{ matrix.python-version }}'
|
||||||
|
run: |
|
||||||
|
make pytest
|
||||||
|
|
||||||
|
- name: 'Upload coverage report'
|
||||||
|
run: bash <(curl -s https://codecov.io/bash)
|
||||||
|
|
||||||
|
- name: 'Run linters'
|
||||||
|
if: matrix.python-version == '3.8'
|
||||||
|
run: |
|
||||||
|
make lint
|
9
Makefile
9
Makefile
|
@ -38,6 +38,15 @@ fix-code-style: ## Fix code formatting
|
||||||
poetry run black --verbose --safe --line-length=${MAX_LINE_LENGTH} --skip-string-normalization .
|
poetry run black --verbose --safe --line-length=${MAX_LINE_LENGTH} --skip-string-normalization .
|
||||||
poetry run isort .
|
poetry run isort .
|
||||||
|
|
||||||
|
tox-listenvs: check-poetry ## List all tox test environments
|
||||||
|
poetry run tox --listenvs
|
||||||
|
|
||||||
|
tox: check-poetry ## Run pytest via tox with all environments
|
||||||
|
poetry run tox
|
||||||
|
|
||||||
|
pytest: install ## Run pytest
|
||||||
|
poetry run python3 ./run_pytest.py
|
||||||
|
|
||||||
local-test: install ## Run local_test.py to run the project locally
|
local-test: install ## Run local_test.py to run the project locally
|
||||||
poetry run python3 ./local_test.py
|
poetry run python3 ./local_test.py
|
||||||
|
|
||||||
|
|
|
@ -12,6 +12,11 @@ django_ynh = "*"
|
||||||
|
|
||||||
[tool.poetry.dev-dependencies]
|
[tool.poetry.dev-dependencies]
|
||||||
bx_py_utils = "*"
|
bx_py_utils = "*"
|
||||||
|
tox = "*"
|
||||||
|
pytest = "*"
|
||||||
|
pytest-cov = "*"
|
||||||
|
pytest-django = "*"
|
||||||
|
coveralls = "*"
|
||||||
isort = "*"
|
isort = "*"
|
||||||
flake8 = "*"
|
flake8 = "*"
|
||||||
flynt = "*"
|
flynt = "*"
|
||||||
|
|
25
run_pytest.py
Normal file
25
run_pytest.py
Normal file
|
@ -0,0 +1,25 @@
|
||||||
|
"""
|
||||||
|
Run pytest against local test creation
|
||||||
|
"""
|
||||||
|
|
||||||
|
from pathlib import Path
|
||||||
|
|
||||||
|
|
||||||
|
try:
|
||||||
|
from django_ynh.pytest_helper import run_pytest
|
||||||
|
except ImportError as err:
|
||||||
|
raise ImportError('Did you forget to activate a virtual environment?') from err
|
||||||
|
|
||||||
|
|
||||||
|
BASE_PATH = Path(__file__).parent
|
||||||
|
|
||||||
|
|
||||||
|
def main():
|
||||||
|
run_pytest(
|
||||||
|
django_settings_path=BASE_PATH / 'conf' / 'settings.py',
|
||||||
|
destination=BASE_PATH / 'local_test',
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
if __name__ == '__main__':
|
||||||
|
main()
|
0
tests/__init__.py
Normal file
0
tests/__init__.py
Normal file
145
tests/test_django_project.py
Normal file
145
tests/test_django_project.py
Normal file
|
@ -0,0 +1,145 @@
|
||||||
|
from axes.models import AccessLog
|
||||||
|
from bx_py_utils.test_utils.html_assertion import HtmlAssertionMixin
|
||||||
|
from django.conf import settings
|
||||||
|
from django.contrib.auth.models import User
|
||||||
|
from django.test import override_settings
|
||||||
|
from django.test.testcases import TestCase
|
||||||
|
from django.urls import NoReverseMatch
|
||||||
|
from django.urls.base import reverse
|
||||||
|
from django_ynh.test_utils import generate_basic_auth
|
||||||
|
from django_ynh.views import request_media_debug_view
|
||||||
|
|
||||||
|
import inventory
|
||||||
|
|
||||||
|
|
||||||
|
@override_settings(DEBUG=False)
|
||||||
|
class DjangoYnhTestCase(HtmlAssertionMixin, TestCase):
|
||||||
|
def setUp(self):
|
||||||
|
super().setUp()
|
||||||
|
|
||||||
|
# Always start a fresh session:
|
||||||
|
self.client = self.client_class()
|
||||||
|
|
||||||
|
def test_settings(self):
|
||||||
|
assert settings.PATH_URL == 'app_path'
|
||||||
|
|
||||||
|
assert str(settings.FINAL_HOME_PATH).endswith('/local_test/opt_yunohost')
|
||||||
|
assert str(settings.FINAL_WWW_PATH).endswith('/local_test/var_www')
|
||||||
|
assert str(settings.LOG_FILE).endswith('/local_test/var_log_django_ynh.log')
|
||||||
|
|
||||||
|
assert settings.ROOT_URLCONF == 'urls'
|
||||||
|
|
||||||
|
def test_urls(self):
|
||||||
|
assert reverse('admin:index') == '/app_path/'
|
||||||
|
|
||||||
|
# The django_ynh debug view should not be avaiable:
|
||||||
|
with self.assertRaises(NoReverseMatch):
|
||||||
|
reverse(request_media_debug_view)
|
||||||
|
|
||||||
|
# Serve user uploads via django_tools.serve_media_app:
|
||||||
|
assert settings.MEDIA_URL == '/app_path/media/'
|
||||||
|
assert reverse('serve_media_app:serve-media', kwargs={'user_token': 'token', 'path': 'foo/bar/'}) == (
|
||||||
|
'/app_path/media/token/foo/bar/'
|
||||||
|
)
|
||||||
|
|
||||||
|
def test_auth(self):
|
||||||
|
response = self.client.get('/app_path/')
|
||||||
|
self.assertRedirects(response, expected_url='/app_path/login/?next=/app_path/')
|
||||||
|
|
||||||
|
def test_create_unknown_user(self):
|
||||||
|
assert User.objects.count() == 0
|
||||||
|
|
||||||
|
self.client.cookies['SSOwAuthUser'] = 'test'
|
||||||
|
|
||||||
|
response = self.client.get(
|
||||||
|
path='/app_path/',
|
||||||
|
HTTP_REMOTE_USER='test',
|
||||||
|
HTTP_AUTH_USER='test',
|
||||||
|
HTTP_AUTHORIZATION='basic dGVzdDp0ZXN0MTIz',
|
||||||
|
)
|
||||||
|
|
||||||
|
assert User.objects.count() == 1
|
||||||
|
user = User.objects.first()
|
||||||
|
assert user.username == 'test'
|
||||||
|
assert user.is_active is True
|
||||||
|
assert user.is_staff is True # Set by: conf.django_ynh_demo_urls.setup_user_handler
|
||||||
|
assert user.is_superuser is False
|
||||||
|
|
||||||
|
self.assert_html_parts(
|
||||||
|
response,
|
||||||
|
parts=(
|
||||||
|
f'<title>Site administration | PyInventory v{inventory.__version__}</title>',
|
||||||
|
'<strong>test</strong>',
|
||||||
|
),
|
||||||
|
)
|
||||||
|
|
||||||
|
def test_wrong_auth_user(self):
|
||||||
|
assert User.objects.count() == 0
|
||||||
|
assert AccessLog.objects.count() == 0
|
||||||
|
|
||||||
|
self.client.cookies['SSOwAuthUser'] = 'test'
|
||||||
|
|
||||||
|
response = self.client.get(
|
||||||
|
path='/app_path/',
|
||||||
|
HTTP_REMOTE_USER='test',
|
||||||
|
HTTP_AUTH_USER='foobar', # <<< wrong user name
|
||||||
|
HTTP_AUTHORIZATION='basic dGVzdDp0ZXN0MTIz',
|
||||||
|
)
|
||||||
|
|
||||||
|
assert User.objects.count() == 1
|
||||||
|
user = User.objects.first()
|
||||||
|
assert user.username == 'test'
|
||||||
|
assert user.is_active is True
|
||||||
|
assert user.is_staff is True # Set by: conf.django_ynh_demo_urls.setup_user_handler
|
||||||
|
assert user.is_superuser is False
|
||||||
|
|
||||||
|
assert AccessLog.objects.count() == 1
|
||||||
|
|
||||||
|
assert response.status_code == 403 # Forbidden
|
||||||
|
|
||||||
|
def test_wrong_cookie(self):
|
||||||
|
assert User.objects.count() == 0
|
||||||
|
assert AccessLog.objects.count() == 0
|
||||||
|
|
||||||
|
self.client.cookies['SSOwAuthUser'] = 'foobar' # <<< wrong user name
|
||||||
|
|
||||||
|
response = self.client.get(
|
||||||
|
path='/app_path/',
|
||||||
|
HTTP_REMOTE_USER='test',
|
||||||
|
HTTP_AUTH_USER='test',
|
||||||
|
HTTP_AUTHORIZATION='basic dGVzdDp0ZXN0MTIz',
|
||||||
|
)
|
||||||
|
|
||||||
|
assert User.objects.count() == 1
|
||||||
|
user = User.objects.first()
|
||||||
|
assert user.username == 'test'
|
||||||
|
assert user.is_active is True
|
||||||
|
assert user.is_staff is True # Set by: conf.django_ynh_demo_urls.setup_user_handler
|
||||||
|
assert user.is_superuser is False
|
||||||
|
|
||||||
|
assert AccessLog.objects.count() == 1
|
||||||
|
|
||||||
|
assert response.status_code == 403 # Forbidden
|
||||||
|
|
||||||
|
def test_wrong_authorization_user(self):
|
||||||
|
assert User.objects.count() == 0
|
||||||
|
|
||||||
|
self.client.cookies['SSOwAuthUser'] = 'test'
|
||||||
|
|
||||||
|
response = self.client.get(
|
||||||
|
path='/app_path/',
|
||||||
|
HTTP_REMOTE_USER='test',
|
||||||
|
HTTP_AUTH_USER='test',
|
||||||
|
HTTP_AUTHORIZATION=generate_basic_auth(username='foobar', password='test123'), # <<< wrong user name
|
||||||
|
)
|
||||||
|
|
||||||
|
assert User.objects.count() == 1
|
||||||
|
user = User.objects.first()
|
||||||
|
assert user.username == 'test'
|
||||||
|
assert user.is_active is True
|
||||||
|
assert user.is_staff is True # Set by: conf.django_ynh_demo_urls.setup_user_handler
|
||||||
|
assert user.is_superuser is False
|
||||||
|
|
||||||
|
assert AccessLog.objects.count() == 1
|
||||||
|
|
||||||
|
assert response.status_code == 403 # Forbidden
|
13
tests/test_lint.py
Normal file
13
tests/test_lint.py
Normal file
|
@ -0,0 +1,13 @@
|
||||||
|
import shutil
|
||||||
|
import subprocess
|
||||||
|
from pathlib import Path
|
||||||
|
|
||||||
|
|
||||||
|
BASE_PATH = Path(__file__).parent.parent
|
||||||
|
|
||||||
|
|
||||||
|
def test_lint():
|
||||||
|
assert Path(BASE_PATH, 'Makefile').is_file()
|
||||||
|
make_bin = shutil.which('make')
|
||||||
|
assert make_bin is not None
|
||||||
|
subprocess.check_call([make_bin, 'lint'], cwd=BASE_PATH)
|
43
tests/test_project_setup.py
Normal file
43
tests/test_project_setup.py
Normal file
|
@ -0,0 +1,43 @@
|
||||||
|
import os
|
||||||
|
import shutil
|
||||||
|
import subprocess
|
||||||
|
from pathlib import Path
|
||||||
|
|
||||||
|
import inventory
|
||||||
|
|
||||||
|
|
||||||
|
PACKAGE_ROOT = Path(__file__).parent.parent
|
||||||
|
|
||||||
|
|
||||||
|
def assert_file_contains_string(file_path, string):
|
||||||
|
with file_path.open('r') as f:
|
||||||
|
for line in f:
|
||||||
|
if string in line:
|
||||||
|
return
|
||||||
|
raise AssertionError(f'File {file_path} does not contain {string!r} !')
|
||||||
|
|
||||||
|
|
||||||
|
def test_version():
|
||||||
|
version = inventory.__version__
|
||||||
|
|
||||||
|
if 'dev' not in version and 'rc' not in version:
|
||||||
|
version_string = f'v{version}'
|
||||||
|
|
||||||
|
assert_file_contains_string(file_path=Path(PACKAGE_ROOT, 'README.md'), string=version_string)
|
||||||
|
|
||||||
|
assert_file_contains_string(file_path=Path(PACKAGE_ROOT, 'pyproject.toml'), string=f'version = "{version}~ynh')
|
||||||
|
assert_file_contains_string(file_path=Path(PACKAGE_ROOT, 'manifest.json'), string=f'"version": "{version}~ynh')
|
||||||
|
|
||||||
|
|
||||||
|
def test_poetry_check():
|
||||||
|
poerty_bin = shutil.which('poetry')
|
||||||
|
|
||||||
|
output = subprocess.check_output(
|
||||||
|
[poerty_bin, 'check'],
|
||||||
|
universal_newlines=True,
|
||||||
|
env=os.environ,
|
||||||
|
stderr=subprocess.STDOUT,
|
||||||
|
cwd=str(PACKAGE_ROOT),
|
||||||
|
)
|
||||||
|
print(output)
|
||||||
|
assert output == 'All set!\n'
|
8
tests/test_utils.py
Normal file
8
tests/test_utils.py
Normal file
|
@ -0,0 +1,8 @@
|
||||||
|
from unittest.case import TestCase
|
||||||
|
|
||||||
|
from django_ynh.test_utils import generate_basic_auth
|
||||||
|
|
||||||
|
|
||||||
|
class TestUtilsTestCase(TestCase):
|
||||||
|
def test_generate_basic_auth(self):
|
||||||
|
assert generate_basic_auth(username='test', password='test123') == 'basic dGVzdDp0ZXN0MTIz'
|
Loading…
Reference in a new issue