refactor: remove database access from core layer
- Remove db_session parameter from SeriesApp, SerieList, SerieScanner - Move all database operations to AnimeService (service layer) - Add add_series_to_db, contains_in_db methods to AnimeService - Update sync_series_from_data_files to use inline DB operations - Remove obsolete test classes for removed DB methods - Fix pylint issues: add broad-except comments, fix line lengths - Core layer (src/core/) now has zero database imports 722 unit tests pass
This commit is contained in:
@@ -1,9 +1,8 @@
|
||||
"""Tests for SerieScanner class - database and file-based operations."""
|
||||
"""Tests for SerieScanner class - file-based operations."""
|
||||
|
||||
import os
|
||||
import tempfile
|
||||
import warnings
|
||||
from unittest.mock import AsyncMock, MagicMock, patch
|
||||
from unittest.mock import MagicMock, patch
|
||||
|
||||
import pytest
|
||||
|
||||
@@ -38,13 +37,6 @@ def mock_loader():
|
||||
return loader
|
||||
|
||||
|
||||
@pytest.fixture
|
||||
def mock_db_session():
|
||||
"""Create a mock async database session."""
|
||||
session = AsyncMock()
|
||||
return session
|
||||
|
||||
|
||||
@pytest.fixture
|
||||
def sample_serie():
|
||||
"""Create a sample Serie for testing."""
|
||||
@@ -68,18 +60,6 @@ class TestSerieScannerInitialization:
|
||||
assert scanner.loader == mock_loader
|
||||
assert scanner.keyDict == {}
|
||||
|
||||
def test_init_with_db_session(
|
||||
self, temp_directory, mock_loader, mock_db_session
|
||||
):
|
||||
"""Test initialization with database session."""
|
||||
scanner = SerieScanner(
|
||||
temp_directory,
|
||||
mock_loader,
|
||||
db_session=mock_db_session
|
||||
)
|
||||
|
||||
assert scanner._db_session == mock_db_session
|
||||
|
||||
def test_init_empty_path_raises_error(self, mock_loader):
|
||||
"""Test initialization with empty path raises ValueError."""
|
||||
with pytest.raises(ValueError, match="empty"):
|
||||
@@ -91,352 +71,40 @@ class TestSerieScannerInitialization:
|
||||
SerieScanner("/nonexistent/path", mock_loader)
|
||||
|
||||
|
||||
class TestSerieScannerScanDeprecation:
|
||||
"""Test scan() deprecation warning."""
|
||||
class TestSerieScannerScan:
|
||||
"""Test file-based scan operations."""
|
||||
|
||||
def test_scan_raises_deprecation_warning(
|
||||
self, temp_directory, mock_loader
|
||||
):
|
||||
"""Test that scan() raises a deprecation warning."""
|
||||
scanner = SerieScanner(temp_directory, mock_loader)
|
||||
|
||||
with warnings.catch_warnings(record=True) as w:
|
||||
warnings.simplefilter("always")
|
||||
|
||||
# Mock the internal methods to avoid actual scanning
|
||||
with patch.object(scanner, 'get_total_to_scan', return_value=0):
|
||||
with patch.object(
|
||||
scanner, '_SerieScanner__find_mp4_files',
|
||||
return_value=iter([])
|
||||
):
|
||||
scanner.scan()
|
||||
|
||||
# Check deprecation warning was raised
|
||||
assert len(w) >= 1
|
||||
deprecation_warnings = [
|
||||
warning for warning in w
|
||||
if issubclass(warning.category, DeprecationWarning)
|
||||
]
|
||||
assert len(deprecation_warnings) >= 1
|
||||
assert "scan_async()" in str(deprecation_warnings[0].message)
|
||||
|
||||
|
||||
class TestSerieScannerAsyncScan:
|
||||
"""Test async database scanning methods."""
|
||||
|
||||
@pytest.mark.asyncio
|
||||
async def test_scan_async_saves_to_database(
|
||||
self, temp_directory, mock_loader, mock_db_session, sample_serie
|
||||
):
|
||||
"""Test scan_async saves results to database."""
|
||||
scanner = SerieScanner(temp_directory, mock_loader)
|
||||
|
||||
# Mock the internal methods
|
||||
with patch.object(scanner, 'get_total_to_scan', return_value=1):
|
||||
with patch.object(
|
||||
scanner,
|
||||
'_SerieScanner__find_mp4_files',
|
||||
return_value=iter([
|
||||
("Attack on Titan (2013)", ["S01E001.mp4"])
|
||||
])
|
||||
):
|
||||
with patch.object(
|
||||
scanner,
|
||||
'_SerieScanner__read_data_from_file',
|
||||
return_value=sample_serie
|
||||
):
|
||||
with patch.object(
|
||||
scanner,
|
||||
'_SerieScanner__get_missing_episodes_and_season',
|
||||
return_value=({1: [2, 3]}, "aniworld.to")
|
||||
):
|
||||
with patch(
|
||||
'src.server.database.service.AnimeSeriesService'
|
||||
) as mock_service:
|
||||
mock_service.get_by_key = AsyncMock(
|
||||
return_value=None
|
||||
)
|
||||
mock_created = MagicMock()
|
||||
mock_created.id = 1
|
||||
mock_service.create = AsyncMock(
|
||||
return_value=mock_created
|
||||
)
|
||||
|
||||
await scanner.scan_async(mock_db_session)
|
||||
|
||||
# Verify database create was called
|
||||
mock_service.create.assert_called_once()
|
||||
|
||||
@pytest.mark.asyncio
|
||||
async def test_scan_async_updates_existing_series(
|
||||
self, temp_directory, mock_loader, mock_db_session, sample_serie
|
||||
):
|
||||
"""Test scan_async updates existing series in database."""
|
||||
scanner = SerieScanner(temp_directory, mock_loader)
|
||||
|
||||
# Mock existing series in database with different episodes
|
||||
existing = MagicMock()
|
||||
existing.id = 1
|
||||
existing.folder = sample_serie.folder
|
||||
|
||||
# Mock episodes (different from sample_serie)
|
||||
mock_existing_episodes = [
|
||||
MagicMock(season=1, episode_number=5),
|
||||
MagicMock(season=1, episode_number=6),
|
||||
]
|
||||
|
||||
with patch.object(scanner, 'get_total_to_scan', return_value=1):
|
||||
with patch.object(
|
||||
scanner,
|
||||
'_SerieScanner__find_mp4_files',
|
||||
return_value=iter([
|
||||
("Attack on Titan (2013)", ["S01E001.mp4"])
|
||||
])
|
||||
):
|
||||
with patch.object(
|
||||
scanner,
|
||||
'_SerieScanner__read_data_from_file',
|
||||
return_value=sample_serie
|
||||
):
|
||||
with patch.object(
|
||||
scanner,
|
||||
'_SerieScanner__get_missing_episodes_and_season',
|
||||
return_value=({1: [2, 3]}, "aniworld.to")
|
||||
):
|
||||
with patch(
|
||||
'src.server.database.service.AnimeSeriesService'
|
||||
) as mock_service:
|
||||
with patch(
|
||||
'src.server.database.service.EpisodeService'
|
||||
) as mock_ep_service:
|
||||
mock_service.get_by_key = AsyncMock(
|
||||
return_value=existing
|
||||
)
|
||||
mock_service.update = AsyncMock(
|
||||
return_value=existing
|
||||
)
|
||||
mock_ep_service.get_by_series = AsyncMock(
|
||||
return_value=mock_existing_episodes
|
||||
)
|
||||
mock_ep_service.create = AsyncMock()
|
||||
|
||||
await scanner.scan_async(mock_db_session)
|
||||
|
||||
# Verify episodes were created
|
||||
assert mock_ep_service.create.called
|
||||
|
||||
@pytest.mark.asyncio
|
||||
async def test_scan_async_handles_errors_gracefully(
|
||||
self, temp_directory, mock_loader, mock_db_session
|
||||
):
|
||||
"""Test scan_async handles folder processing errors gracefully."""
|
||||
scanner = SerieScanner(temp_directory, mock_loader)
|
||||
|
||||
with patch.object(scanner, 'get_total_to_scan', return_value=1):
|
||||
with patch.object(
|
||||
scanner,
|
||||
'_SerieScanner__find_mp4_files',
|
||||
return_value=iter([
|
||||
("Error Folder", ["S01E001.mp4"])
|
||||
])
|
||||
):
|
||||
with patch.object(
|
||||
scanner,
|
||||
'_SerieScanner__read_data_from_file',
|
||||
side_effect=Exception("Test error")
|
||||
):
|
||||
# Should not raise, should continue
|
||||
await scanner.scan_async(mock_db_session)
|
||||
|
||||
|
||||
class TestSerieScannerDatabaseHelpers:
|
||||
"""Test database helper methods."""
|
||||
|
||||
@pytest.mark.asyncio
|
||||
async def test_save_serie_to_db_creates_new(
|
||||
self, temp_directory, mock_loader, mock_db_session, sample_serie
|
||||
):
|
||||
"""Test _save_serie_to_db creates new series."""
|
||||
scanner = SerieScanner(temp_directory, mock_loader)
|
||||
|
||||
with patch(
|
||||
'src.server.database.service.AnimeSeriesService'
|
||||
) as mock_service:
|
||||
with patch(
|
||||
'src.server.database.service.EpisodeService'
|
||||
) as mock_ep_service:
|
||||
mock_service.get_by_key = AsyncMock(return_value=None)
|
||||
mock_created = MagicMock()
|
||||
mock_created.id = 1
|
||||
mock_service.create = AsyncMock(return_value=mock_created)
|
||||
mock_ep_service.create = AsyncMock()
|
||||
|
||||
result = await scanner._save_serie_to_db(
|
||||
sample_serie, mock_db_session
|
||||
)
|
||||
|
||||
assert result is mock_created
|
||||
mock_service.create.assert_called_once()
|
||||
|
||||
@pytest.mark.asyncio
|
||||
async def test_save_serie_to_db_updates_existing(
|
||||
self, temp_directory, mock_loader, mock_db_session, sample_serie
|
||||
):
|
||||
"""Test _save_serie_to_db updates existing series."""
|
||||
scanner = SerieScanner(temp_directory, mock_loader)
|
||||
|
||||
existing = MagicMock()
|
||||
existing.id = 1
|
||||
existing.folder = sample_serie.folder
|
||||
|
||||
# Mock existing episodes (different from sample_serie)
|
||||
mock_existing_episodes = [
|
||||
MagicMock(season=1, episode_number=5),
|
||||
MagicMock(season=1, episode_number=6),
|
||||
]
|
||||
|
||||
with patch(
|
||||
'src.server.database.service.AnimeSeriesService'
|
||||
) as mock_service:
|
||||
with patch(
|
||||
'src.server.database.service.EpisodeService'
|
||||
) as mock_ep_service:
|
||||
mock_service.get_by_key = AsyncMock(return_value=existing)
|
||||
mock_service.update = AsyncMock(return_value=existing)
|
||||
mock_ep_service.get_by_series = AsyncMock(
|
||||
return_value=mock_existing_episodes
|
||||
)
|
||||
mock_ep_service.create = AsyncMock()
|
||||
|
||||
result = await scanner._save_serie_to_db(
|
||||
sample_serie, mock_db_session
|
||||
)
|
||||
|
||||
assert result is existing
|
||||
# Should have created new episodes
|
||||
assert mock_ep_service.create.called
|
||||
|
||||
@pytest.mark.asyncio
|
||||
async def test_save_serie_to_db_skips_unchanged(
|
||||
self, temp_directory, mock_loader, mock_db_session, sample_serie
|
||||
):
|
||||
"""Test _save_serie_to_db skips update if unchanged."""
|
||||
scanner = SerieScanner(temp_directory, mock_loader)
|
||||
|
||||
existing = MagicMock()
|
||||
existing.id = 1
|
||||
existing.folder = sample_serie.folder
|
||||
|
||||
# Mock episodes matching sample_serie.episodeDict
|
||||
mock_existing_episodes = []
|
||||
for season, ep_nums in sample_serie.episodeDict.items():
|
||||
for ep_num in ep_nums:
|
||||
mock_existing_episodes.append(
|
||||
MagicMock(season=season, episode_number=ep_num)
|
||||
)
|
||||
|
||||
with patch(
|
||||
'src.server.database.service.AnimeSeriesService'
|
||||
) as mock_service:
|
||||
with patch(
|
||||
'src.server.database.service.EpisodeService'
|
||||
) as mock_ep_service:
|
||||
mock_service.get_by_key = AsyncMock(return_value=existing)
|
||||
mock_ep_service.get_by_series = AsyncMock(
|
||||
return_value=mock_existing_episodes
|
||||
)
|
||||
|
||||
result = await scanner._save_serie_to_db(
|
||||
sample_serie, mock_db_session
|
||||
)
|
||||
|
||||
assert result is None
|
||||
mock_service.update.assert_not_called()
|
||||
|
||||
@pytest.mark.asyncio
|
||||
async def test_update_serie_in_db_updates_existing(
|
||||
self, temp_directory, mock_loader, mock_db_session, sample_serie
|
||||
):
|
||||
"""Test _update_serie_in_db updates existing series."""
|
||||
scanner = SerieScanner(temp_directory, mock_loader)
|
||||
|
||||
existing = MagicMock()
|
||||
existing.id = 1
|
||||
|
||||
with patch(
|
||||
'src.server.database.service.AnimeSeriesService'
|
||||
) as mock_service:
|
||||
with patch(
|
||||
'src.server.database.service.EpisodeService'
|
||||
) as mock_ep_service:
|
||||
mock_service.get_by_key = AsyncMock(return_value=existing)
|
||||
mock_service.update = AsyncMock(return_value=existing)
|
||||
mock_ep_service.get_by_series = AsyncMock(return_value=[])
|
||||
mock_ep_service.create = AsyncMock()
|
||||
|
||||
result = await scanner._update_serie_in_db(
|
||||
sample_serie, mock_db_session
|
||||
)
|
||||
|
||||
assert result is existing
|
||||
mock_service.update.assert_called_once()
|
||||
|
||||
@pytest.mark.asyncio
|
||||
async def test_update_serie_in_db_returns_none_if_not_found(
|
||||
self, temp_directory, mock_loader, mock_db_session, sample_serie
|
||||
):
|
||||
"""Test _update_serie_in_db returns None if series not found."""
|
||||
scanner = SerieScanner(temp_directory, mock_loader)
|
||||
|
||||
with patch(
|
||||
'src.server.database.service.AnimeSeriesService'
|
||||
) as mock_service:
|
||||
mock_service.get_by_key = AsyncMock(return_value=None)
|
||||
|
||||
result = await scanner._update_serie_in_db(
|
||||
sample_serie, mock_db_session
|
||||
)
|
||||
|
||||
assert result is None
|
||||
|
||||
|
||||
class TestSerieScannerBackwardCompatibility:
|
||||
"""Test backward compatibility of file-based operations."""
|
||||
|
||||
def test_file_based_scan_still_works(
|
||||
def test_file_based_scan_works(
|
||||
self, temp_directory, mock_loader, sample_serie
|
||||
):
|
||||
"""Test file-based scan still works with deprecation warning."""
|
||||
"""Test file-based scan works properly."""
|
||||
scanner = SerieScanner(temp_directory, mock_loader)
|
||||
|
||||
with warnings.catch_warnings():
|
||||
warnings.simplefilter("ignore", DeprecationWarning)
|
||||
|
||||
with patch.object(scanner, 'get_total_to_scan', return_value=1):
|
||||
with patch.object(scanner, 'get_total_to_scan', return_value=1):
|
||||
with patch.object(
|
||||
scanner,
|
||||
'_SerieScanner__find_mp4_files',
|
||||
return_value=iter([
|
||||
("Attack on Titan (2013)", ["S01E001.mp4"])
|
||||
])
|
||||
):
|
||||
with patch.object(
|
||||
scanner,
|
||||
'_SerieScanner__find_mp4_files',
|
||||
return_value=iter([
|
||||
("Attack on Titan (2013)", ["S01E001.mp4"])
|
||||
])
|
||||
'_SerieScanner__read_data_from_file',
|
||||
return_value=sample_serie
|
||||
):
|
||||
with patch.object(
|
||||
scanner,
|
||||
'_SerieScanner__read_data_from_file',
|
||||
return_value=sample_serie
|
||||
'_SerieScanner__get_missing_episodes_and_season',
|
||||
return_value=({1: [2, 3]}, "aniworld.to")
|
||||
):
|
||||
with patch.object(
|
||||
scanner,
|
||||
'_SerieScanner__get_missing_episodes_and_season',
|
||||
return_value=({1: [2, 3]}, "aniworld.to")
|
||||
):
|
||||
with patch.object(
|
||||
sample_serie, 'save_to_file'
|
||||
) as mock_save:
|
||||
scanner.scan()
|
||||
|
||||
# Verify file was saved
|
||||
mock_save.assert_called_once()
|
||||
sample_serie, 'save_to_file'
|
||||
) as mock_save:
|
||||
scanner.scan()
|
||||
|
||||
# Verify file was saved
|
||||
mock_save.assert_called_once()
|
||||
|
||||
def test_keydict_populated_after_scan(
|
||||
self, temp_directory, mock_loader, sample_serie
|
||||
@@ -444,28 +112,25 @@ class TestSerieScannerBackwardCompatibility:
|
||||
"""Test keyDict is populated after scan."""
|
||||
scanner = SerieScanner(temp_directory, mock_loader)
|
||||
|
||||
with warnings.catch_warnings():
|
||||
warnings.simplefilter("ignore", DeprecationWarning)
|
||||
|
||||
with patch.object(scanner, 'get_total_to_scan', return_value=1):
|
||||
with patch.object(scanner, 'get_total_to_scan', return_value=1):
|
||||
with patch.object(
|
||||
scanner,
|
||||
'_SerieScanner__find_mp4_files',
|
||||
return_value=iter([
|
||||
("Attack on Titan (2013)", ["S01E001.mp4"])
|
||||
])
|
||||
):
|
||||
with patch.object(
|
||||
scanner,
|
||||
'_SerieScanner__find_mp4_files',
|
||||
return_value=iter([
|
||||
("Attack on Titan (2013)", ["S01E001.mp4"])
|
||||
])
|
||||
'_SerieScanner__read_data_from_file',
|
||||
return_value=sample_serie
|
||||
):
|
||||
with patch.object(
|
||||
scanner,
|
||||
'_SerieScanner__read_data_from_file',
|
||||
return_value=sample_serie
|
||||
'_SerieScanner__get_missing_episodes_and_season',
|
||||
return_value=({1: [2, 3]}, "aniworld.to")
|
||||
):
|
||||
with patch.object(
|
||||
scanner,
|
||||
'_SerieScanner__get_missing_episodes_and_season',
|
||||
return_value=({1: [2, 3]}, "aniworld.to")
|
||||
):
|
||||
with patch.object(sample_serie, 'save_to_file'):
|
||||
scanner.scan()
|
||||
|
||||
assert sample_serie.key in scanner.keyDict
|
||||
with patch.object(sample_serie, 'save_to_file'):
|
||||
scanner.scan()
|
||||
|
||||
assert sample_serie.key in scanner.keyDict
|
||||
|
||||
Reference in New Issue
Block a user