Files
main_backend/tests/unit/test_task_service.py
2026-04-27 23:19:04 +03:00

230 lines
7.5 KiB
Python

from __future__ import annotations
import uuid
from types import SimpleNamespace
from unittest.mock import AsyncMock
import pytest
from cpv3.modules.tasks.schemas import CaptionsGenerateRequest, TaskWebhookEvent
from cpv3.modules.tasks.service import TaskService
@pytest.mark.asyncio
async def test_submit_captions_generate_reuses_existing_active_job() -> None:
service = TaskService(session=AsyncMock())
existing_job_id = uuid.uuid4()
existing_job = SimpleNamespace(
id=existing_job_id,
status="RUNNING",
)
service._find_duplicate_active_job = AsyncMock(return_value=existing_job)
service._submit_task = AsyncMock()
response = await service.submit_captions_generate(
requester=SimpleNamespace(id=uuid.uuid4()),
request=CaptionsGenerateRequest(
video_s3_path="projects/test/video.mp4",
folder="output_files",
transcription_id=uuid.uuid4(),
project_id=uuid.uuid4(),
preset_id=uuid.uuid4(),
),
)
assert response.job_id == existing_job_id
assert response.status == "RUNNING"
assert response.webhook_url.endswith(f"/api/tasks/webhook/{existing_job_id}/")
service._submit_task.assert_not_awaited()
@pytest.mark.asyncio
async def test_record_webhook_event_ignores_cancelled_job() -> None:
cancelled_job = SimpleNamespace(
id=uuid.uuid4(),
status="CANCELLED",
)
job_repo = SimpleNamespace(
get_by_id=AsyncMock(return_value=cancelled_job),
update=AsyncMock(),
)
event_repo = SimpleNamespace(create=AsyncMock())
service = TaskService(session=AsyncMock())
service._job_repo = job_repo
service._event_repo = event_repo
result = await service.record_webhook_event(
job_id=cancelled_job.id,
event=TaskWebhookEvent(
status="DONE",
current_message="Готово",
output_data={"output_path": "projects/test/output.mp4"},
),
)
assert result is cancelled_job
job_repo.update.assert_not_awaited()
event_repo.create.assert_not_awaited()
@pytest.mark.asyncio
async def test_cancel_job_marks_job_cancelled_and_keeps_record() -> None:
job_id = uuid.uuid4()
user_id = uuid.uuid4()
job = SimpleNamespace(
id=job_id,
status="PENDING",
broker_id="default:redis-message-id",
job_type="CAPTIONS_GENERATE",
user_id=user_id,
)
cancelled_job = SimpleNamespace(
id=job_id,
status="CANCELLED",
broker_id="default:redis-message-id",
job_type="CAPTIONS_GENERATE",
user_id=user_id,
current_message="Отменено пользователем",
)
service = TaskService(session=AsyncMock())
service._job_repo = SimpleNamespace(update=AsyncMock(return_value=cancelled_job))
service._event_repo = SimpleNamespace(create=AsyncMock())
service._cancel_dramatiq_message = AsyncMock()
service._cancel_caption_render = AsyncMock()
service._create_cancellation_notification = AsyncMock()
service._sync_project_workspace_after_webhook = AsyncMock()
result = await service.cancel_job(job)
assert result is cancelled_job
service._job_repo.update.assert_awaited_once()
service._event_repo.create.assert_awaited_once()
service._cancel_dramatiq_message.assert_awaited_once_with(job.broker_id)
service._cancel_caption_render.assert_awaited_once_with(job)
service._create_cancellation_notification.assert_awaited_once_with(cancelled_job)
service._sync_project_workspace_after_webhook.assert_awaited_once_with(cancelled_job)
@pytest.mark.asyncio
async def test_record_webhook_event_updates_progress_for_conversion_job() -> None:
job = SimpleNamespace(
id=uuid.uuid4(),
status="RUNNING",
job_type="MEDIA_CONVERT",
project_id=uuid.uuid4(),
user_id=None,
)
updated_job = SimpleNamespace(**job.__dict__, project_pct=52.5)
job_repo = SimpleNamespace(
get_by_id=AsyncMock(return_value=job),
update=AsyncMock(return_value=updated_job),
)
event_repo = SimpleNamespace(create=AsyncMock())
service = TaskService(session=AsyncMock())
service._job_repo = job_repo
service._event_repo = event_repo
service._sync_project_workspace_after_webhook = AsyncMock()
result = await service.record_webhook_event(
job_id=job.id,
event=TaskWebhookEvent(
progress_pct=52.5,
current_message="Конвертация видео",
),
)
update_call = job_repo.update.await_args.args[1]
event_call = event_repo.create.await_args.args[0]
assert result is updated_job
assert update_call.project_pct == 52.5
assert update_call.current_message == "Конвертация видео"
assert event_call.event_type == "progress"
assert event_call.payload["progress_pct"] == 52.5
@pytest.mark.asyncio
async def test_record_webhook_event_syncs_workspace_for_completed_supported_job() -> None:
job = SimpleNamespace(
id=uuid.uuid4(),
status="RUNNING",
job_type="SILENCE_DETECT",
user_id=None,
project_id=uuid.uuid4(),
output_data={"silent_segments": []},
)
updated_job = SimpleNamespace(**{**job.__dict__, "status": "DONE"})
job_repo = SimpleNamespace(
get_by_id=AsyncMock(return_value=job),
update=AsyncMock(return_value=updated_job),
)
event_repo = SimpleNamespace(create=AsyncMock())
service = TaskService(session=AsyncMock())
service._job_repo = job_repo
service._event_repo = event_repo
service._sync_project_workspace_after_webhook = AsyncMock()
result = await service.record_webhook_event(
job_id=job.id,
event=TaskWebhookEvent(
status="DONE",
current_message="Готово",
output_data={"silent_segments": []},
),
)
assert result is updated_job
service._sync_project_workspace_after_webhook.assert_awaited_once_with(updated_job)
@pytest.mark.asyncio
async def test_record_webhook_event_projects_workspace_after_done_job() -> None:
job = SimpleNamespace(
id=uuid.uuid4(),
status="RUNNING",
job_type="MEDIA_CONVERT",
project_id=uuid.uuid4(),
user_id=None,
output_data={"file_path": "users/test/converted.mp4"},
)
updated_job = SimpleNamespace(
**{
**job.__dict__,
"status": "DONE",
"output_data": {
"file_path": "users/test/converted.mp4",
"file_id": "00000000-0000-4000-a000-000000000777",
},
}
)
job_repo = SimpleNamespace(
get_by_id=AsyncMock(return_value=job),
update=AsyncMock(return_value=updated_job),
)
event_repo = SimpleNamespace(create=AsyncMock())
workspace_service = SimpleNamespace(handle_job_update=AsyncMock())
service = TaskService(session=AsyncMock())
service._job_repo = job_repo
service._event_repo = event_repo
service._save_convert_artifacts = AsyncMock(return_value=updated_job)
service._get_project_workspace_service = lambda: workspace_service
result = await service.record_webhook_event(
job_id=job.id,
event=TaskWebhookEvent(
status="DONE",
current_message="Готово",
output_data={"file_path": "users/test/converted.mp4"},
),
)
assert result is updated_job
service._save_convert_artifacts.assert_awaited_once_with(updated_job)
workspace_service.handle_job_update.assert_awaited_once_with(job=updated_job)