Files
archived-MoviePilot/tests/test_media_recognize_modules.py
2026-05-20 10:55:01 +08:00

222 lines
8.5 KiB
Python
Raw Blame History

This file contains ambiguous Unicode characters
This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.
import asyncio
from unittest import TestCase
from unittest.mock import AsyncMock, Mock, patch
from app.core.context import MediaInfo
from app.core.meta import MetaBase
from app.modules.douban import DoubanModule
from app.modules.themoviedb import TheMovieDbModule
from app.modules.themoviedb.scraper import TmdbScraper
from app.modules.themoviedb.tmdbapi import TmdbApi
from app.schemas.types import MediaType
class MediaRecognizeModulesTest(TestCase):
def test_tmdb_cache_false_skips_cache_lookup(self):
"""cache=False 时应跳过缓存读取,但仍按正常流程查询 TMDB。"""
module = TheMovieDbModule()
meta = MetaBase("测试电影")
meta.name = "测试电影"
meta.type = MediaType.MOVIE
module.cache = Mock()
module.tmdb = Mock()
module.tmdb.get_info.return_value = {
"id": 100,
"media_type": MediaType.MOVIE,
"title": "测试电影",
"genres": [],
}
module.category = Mock()
module.category.get_movie_category.return_value = None
result = module.recognize_media(meta=meta, tmdbid=100, cache=False)
self.assertIsInstance(result, MediaInfo)
self.assertEqual(result.tmdb_id, 100)
module.cache.get.assert_not_called()
module.cache.update.assert_called_once()
def test_async_tmdb_cache_false_skips_cache_lookup(self):
"""异步 cache=False 时也应跳过缓存读取。"""
module = TheMovieDbModule()
meta = MetaBase("测试电影")
meta.name = "测试电影"
meta.type = MediaType.MOVIE
module.cache = Mock()
module.tmdb = Mock()
async def _async_get_info(**kwargs):
return {
"id": 101,
"media_type": MediaType.MOVIE,
"title": "测试电影",
"genres": [],
}
module.tmdb.async_get_info = _async_get_info
module.category = Mock()
module.category.get_movie_category.return_value = None
result = asyncio.run(module.async_recognize_media(meta=meta, tmdbid=101, cache=False))
self.assertIsInstance(result, MediaInfo)
self.assertEqual(result.tmdb_id, 101)
module.cache.get.assert_not_called()
module.cache.update.assert_called_once()
def test_tmdb_recognize_does_not_fallback_to_match_web(self):
"""TMDB API 搜索无结果时,不应再回退抓取 TMDB 网站搜索页。"""
module = TheMovieDbModule()
meta = MetaBase("No Match Movie")
meta.name = "No Match Movie"
meta.type = MediaType.MOVIE
module.cache = Mock()
module.tmdb = Mock()
module.tmdb.match_web.side_effect = AssertionError("不应调用 TMDB 网站搜索")
module._search_by_name = Mock(return_value=None)
result = module.recognize_media(meta=meta, cache=False)
self.assertIsNone(result)
module._search_by_name.assert_called()
module.tmdb.match_web.assert_not_called()
def test_async_tmdb_recognize_does_not_fallback_to_match_web(self):
"""异步 TMDB API 搜索无结果时,不应再回退抓取 TMDB 网站搜索页。"""
module = TheMovieDbModule()
meta = MetaBase("No Match Movie")
meta.name = "No Match Movie"
meta.type = MediaType.MOVIE
module.cache = Mock()
module.tmdb = Mock()
module.tmdb.async_match_web = AsyncMock(side_effect=AssertionError("不应调用 TMDB 网站搜索"))
module._async_search_by_name = AsyncMock(return_value=None)
result = asyncio.run(module.async_recognize_media(meta=meta, cache=False))
self.assertIsNone(result)
module._async_search_by_name.assert_called()
module.tmdb.async_match_web.assert_not_called()
def test_tmdb_image_language_fallback_includes_current_en_null_and_original(self):
"""TMDB 图片查询应带上语言回退,避免当前语言没有图片时直接返回空。"""
with patch("app.modules.themoviedb.tmdbapi.settings") as mock_settings:
mock_settings.TMDB_LOCALE = "zh"
result = TmdbApi._build_include_image_language("ja")
self.assertEqual(result, "zh,en,null,ja")
def test_tmdb_trending_filters_non_media_and_normalizes_media_type(self):
"""TMDB流行趋势应过滤人物项并把字符串媒体类型转为内部枚举。"""
infos = [
{"id": 100, "media_type": "movie", "title": "测试电影"},
{"id": 101, "media_type": "tv", "name": "测试剧集"},
{"id": 102, "media_type": "person", "name": "测试人物"},
{"id": 103, "media_type": MediaType.MOVIE, "title": "枚举电影"},
]
result = TmdbApi._normalize_trending_infos(infos)
self.assertEqual([info["id"] for info in result], [100, 101, 103])
self.assertEqual(result[0]["media_type"], MediaType.MOVIE)
self.assertEqual(result[1]["media_type"], MediaType.TV)
self.assertEqual(result[2]["media_type"], MediaType.MOVIE)
def test_tmdb_obtain_images_uses_language_fallback_and_picks_best(self):
"""obtain_images 应从图片接口回填缺失的海报和背景图。"""
module = TheMovieDbModule()
module.tmdb = Mock()
module.tmdb.get_movie_images.return_value = {
"posters": [
{"file_path": "/low-poster.jpg", "vote_average": 2, "vote_count": 10},
{"file_path": "/best-poster.jpg", "vote_average": 8, "vote_count": 1},
],
"backdrops": [
{"file_path": "/best-backdrop.jpg", "vote_average": 7, "vote_count": 2},
],
}
mediainfo = MediaInfo(
tmdb_id=100,
type=MediaType.MOVIE,
original_language="ja",
)
result = module.obtain_images(mediainfo)
self.assertIs(result, mediainfo)
module.tmdb.get_movie_images.assert_called_once_with(100, original_language="ja")
self.assertTrue(mediainfo.poster_path.endswith("/best-poster.jpg"))
self.assertTrue(mediainfo.backdrop_path.endswith("/best-backdrop.jpg"))
def test_tmdb_scraper_metadata_img_fetches_missing_main_images(self):
"""主媒体图片缺失时,刮削图片列表应先从 TMDB images 接口补齐。"""
scraper = TmdbScraper()
scraper._meta_tmdb = Mock()
scraper._meta_tmdb.get_movie_images.return_value = {
"posters": [
{"file_path": "/fallback-poster.jpg", "vote_average": 5},
],
"backdrops": [
{"file_path": "/fallback-backdrop.jpg", "vote_average": 4},
],
}
mediainfo = MediaInfo(
tmdb_id=200,
type=MediaType.MOVIE,
original_language="en",
)
images = scraper.get_metadata_img(mediainfo)
scraper._meta_tmdb.get_movie_images.assert_called_once_with(
200,
original_language="en",
)
self.assertIn("poster.jpg", images)
self.assertIn("backdrop.jpg", images)
def test_douban_prepare_search_names_deduplicates_simplified_name(self):
"""豆瓣候选名称应保留顺序,并去掉繁简转换后的重复项。"""
meta = MetaBase("流浪地球")
meta.cn_name = "流浪地球"
meta.en_name = "The Wandering Earth"
self.assertEqual(
DoubanModule._prepare_search_names(meta),
["流浪地球", "The Wandering Earth"],
)
def test_douban_search_result_helper_preserves_season_title_rule(self):
"""豆瓣搜索结果 helper 应保留电视剧标题追加季号的旧逻辑。"""
meta = MetaBase("测试剧")
meta.name = "测试剧"
meta.type = MediaType.TV
meta.begin_season = 2
items = [
{
"type_name": MediaType.TV.value,
"target": {
"id": "200",
"title": "测试剧",
"type": "tv",
"year": "2024",
},
},
{
"type_name": MediaType.MOVIE.value,
"target": {
"id": "201",
"title": "测试剧 电影版",
"type": "movie",
"year": "2024",
},
},
]
result = DoubanModule._build_search_medias_result(meta, items)
self.assertEqual(len(result), 1)
self.assertEqual(result[0].title, "测试剧 第二季")
self.assertEqual(result[0].season, 2)