test: add unit tests for roles, web deps, use cases, VO boundaries — reach 70% coverage
Some checks failed
ci/woodpecker/pr/pipeline Pipeline failed

This commit is contained in:
2026-05-09 12:57:25 +03:00
parent 2b8a5676bd
commit 79f4d9caf5
12 changed files with 466 additions and 8 deletions

View File

@@ -1,5 +1,6 @@
"""Tests for domain entities."""
from typing import Any
from uuid import UUID
from app.domain.entities import Post
@@ -126,3 +127,19 @@ class TestPost:
assert "id" in data
assert "created_at" in data
assert "updated_at" in data
def test_base_entity_eq_and_hash(self) -> None:
"""Test BaseEntity equality and hash directly."""
from app.domain.entities.base import BaseEntity
class ConcreteEntity(BaseEntity):
def to_dict(self) -> dict[str, Any]:
return {}
e1 = ConcreteEntity()
e2 = ConcreteEntity()
e2.id = e1.id
assert BaseEntity.__eq__(e1, e2) is True
assert BaseEntity.__eq__(e1, "not an entity") == NotImplemented
assert BaseEntity.__hash__(e1) == hash(e1.id)

View File

@@ -1,11 +1,17 @@
"""Tests for role-based access control."""
from typing import Any
import pytest
from app.domain.exceptions import ForbiddenException
from app.domain.roles import (
ROLE_PERMISSIONS,
Permission,
Role,
get_effective_role,
has_permission,
require_permission,
)
@@ -121,3 +127,61 @@ class TestGetEffectiveRole:
assert get_effective_role(["user", "admin", "guest"]) == Role.ADMIN
# User takes precedence over guest
assert get_effective_role(["guest", "user"]) == Role.USER
class TestRequirePermission:
"""Test require_permission decorator."""
async def test_admin_passes_permission_check(self) -> None:
"""Test admin with permission succeeds."""
@require_permission(Permission.POST_CREATE)
async def dummy(*, token_info: Any = None) -> str:
return "ok"
token = type("TokenInfo", (), {"roles": ["admin"]})()
result = await dummy(token_info=token)
assert result == "ok"
async def test_user_passes_permission_check(self) -> None:
"""Test user with permission succeeds."""
@require_permission(Permission.POST_READ)
async def dummy(*, token_info: Any = None) -> str:
return "ok"
token = type("TokenInfo", (), {"roles": ["user"]})()
result = await dummy(token_info=token)
assert result == "ok"
async def test_no_token_raises(self) -> None:
"""Test missing token raises ForbiddenException."""
@require_permission(Permission.POST_CREATE)
async def dummy(*, token_info: Any = None) -> str:
return "ok"
with pytest.raises(ForbiddenException, match="Authentication required"):
await dummy()
async def test_guest_without_permission_raises(self) -> None:
"""Test guest lacking permission raises ForbiddenException."""
@require_permission(Permission.POST_CREATE)
async def dummy(*, token_info: Any = None) -> str:
return "ok"
token = type("TokenInfo", (), {"roles": []})()
with pytest.raises(ForbiddenException, match="Permission 'post:create' required"):
await dummy(token_info=token)
async def test_user_forbidden_unpublished(self) -> None:
"""Test user cannot read unpublished via decorator."""
@require_permission(Permission.POST_READ_UNPUBLISHED)
async def dummy(*, token_info: Any = None) -> str:
return "ok"
token = type("TokenInfo", (), {"roles": ["user"]})()
with pytest.raises(ForbiddenException, match="Permission 'post:read_unpublished' required"):
await dummy(token_info=token)

View File

@@ -91,3 +91,73 @@ class TestSlug:
slug2 = Slug("test-slug")
assert slug1 == slug2
assert hash(slug1) == hash(slug2)
def test_slug_str_and_primitive(self) -> None:
"""Test slug string and primitive conversion."""
slug = Slug("test-slug")
assert str(slug) == "test-slug"
assert slug.to_primitive() == "test-slug"
def test_slug_eq_with_non_value_object(self) -> None:
"""Test slug equality with non-ValueObject returns False."""
slug = Slug("test-slug")
assert slug != "test-slug"
def test_slug_not_string(self) -> None:
"""Test non-string slug raises ValueError."""
with pytest.raises(ValueError, match="string"):
Slug(123) # type: ignore[arg-type]
def test_slug_too_long(self) -> None:
"""Test slug exceeding max length raises ValueError."""
with pytest.raises(ValueError, match="at most"):
Slug("a" * 201)
class TestValueObjectBase:
"""Test base ValueObject behavior across types."""
def test_title_eq_with_non_value_object(self) -> None:
"""Test title equality with non-ValueObject returns False."""
title = Title("Test Title")
assert title != "Test Title"
def test_content_str_and_primitive(self) -> None:
"""Test content string and primitive conversion."""
content = Content("Enough characters for valid content")
assert str(content) == "Enough characters for valid content"
assert content.to_primitive() == "Enough characters for valid content"
def test_content_not_string(self) -> None:
"""Test non-string content raises ValueError."""
with pytest.raises(ValueError, match="string"):
Content(123) # type: ignore[arg-type]
def test_value_object_eq_and_hash_directly(self) -> None:
"""Test ValueObject base eq and hash via direct call."""
from app.domain.value_objects.base import ValueObject
class SimpleVO(ValueObject[str]):
def _validate(self) -> None:
pass
vo1 = SimpleVO("test")
vo2 = SimpleVO("test")
vo3 = SimpleVO("other")
assert ValueObject.__eq__(vo1, vo2) is True
assert ValueObject.__eq__(vo1, vo3) is False
assert ValueObject.__eq__(vo1, "test") is False
assert ValueObject.__hash__(vo1) == hash("test")
def test_content_exact_min_length(self) -> None:
"""Test content at exact minimum length boundary."""
min_content = "a" * 10
content = Content(min_content)
assert content.value == min_content
def test_slug_exact_max_length(self) -> None:
"""Test slug at exact maximum length boundary."""
max_slug = "a" * 200
slug = Slug(max_slug)
assert slug.value == max_slug