mirror of
https://github.com/NousResearch/hermes-agent.git
synced 2026-07-01 12:02:05 +00:00
Add durable public-URL output and URL-based chaining to xAI Grok Imagine: - Store generated media on files-cdn with permanent public HTTPS URLs (public_url: true, no expiry by default). - Chain by URL: generate -> edit -> extend each take a prior result's public HTTPS URL (or a data URI / local file for inputs). - Add provider-specific xai_video_edit and xai_video_extend tools. - Image generation: public-URL/storage output, multi-reference edits, and ~/ local-path support for image edits. Credentials use xAI Grok device-code OAuth (separate PR).
140 lines
5.1 KiB
Python
140 lines
5.1 KiB
Python
"""Tests for the unified ``video_generate`` tool dispatch surface."""
|
|
|
|
from __future__ import annotations
|
|
|
|
import json
|
|
from typing import Any, Dict, List, Optional
|
|
|
|
import pytest
|
|
|
|
from agent import video_gen_registry
|
|
from agent.video_gen_provider import VideoGenProvider
|
|
|
|
|
|
@pytest.fixture(autouse=True)
|
|
def _reset_registry():
|
|
video_gen_registry._reset_for_tests()
|
|
yield
|
|
video_gen_registry._reset_for_tests()
|
|
|
|
|
|
class _RecordingProvider(VideoGenProvider):
|
|
"""Captures the kwargs the tool layer hands it."""
|
|
|
|
def __init__(self, name: str = "fake"):
|
|
self._name = name
|
|
self.last_kwargs: Dict[str, Any] = {}
|
|
|
|
@property
|
|
def name(self) -> str:
|
|
return self._name
|
|
|
|
def list_models(self) -> List[Dict[str, Any]]:
|
|
return [{"id": "model-a"}]
|
|
|
|
def default_model(self) -> Optional[str]:
|
|
return "model-a"
|
|
|
|
def capabilities(self) -> Dict[str, Any]:
|
|
return {"modalities": ["text", "image"]}
|
|
|
|
def generate(self, prompt, **kwargs):
|
|
self.last_kwargs = {"prompt": prompt, **kwargs}
|
|
modality = "image" if kwargs.get("image_url") else "text"
|
|
return {
|
|
"success": True,
|
|
"video": "https://example.com/v.mp4",
|
|
"model": kwargs.get("model") or "model-a",
|
|
"prompt": prompt,
|
|
"modality": modality,
|
|
"aspect_ratio": kwargs.get("aspect_ratio", ""),
|
|
"duration": kwargs.get("duration") or 0,
|
|
"provider": self._name,
|
|
}
|
|
|
|
|
|
class _RaisingProvider(VideoGenProvider):
|
|
@property
|
|
def name(self) -> str:
|
|
return "raises"
|
|
|
|
def generate(self, prompt, **kwargs):
|
|
raise RuntimeError("boom")
|
|
|
|
|
|
class TestUnifiedDispatch:
|
|
def _run(self, args: Dict[str, Any], *, configured: Optional[str] = None) -> Dict[str, Any]:
|
|
from tools import video_generation_tool
|
|
import hermes_cli.plugins as plugins_module
|
|
|
|
saved = video_generation_tool._read_configured_video_provider
|
|
video_generation_tool._read_configured_video_provider = lambda: configured # type: ignore
|
|
saved_discover = plugins_module._ensure_plugins_discovered
|
|
plugins_module._ensure_plugins_discovered = lambda *_a, **_k: None # type: ignore
|
|
try:
|
|
raw = video_generation_tool._handle_video_generate(args)
|
|
finally:
|
|
video_generation_tool._read_configured_video_provider = saved # type: ignore
|
|
plugins_module._ensure_plugins_discovered = saved_discover # type: ignore
|
|
return json.loads(raw)
|
|
|
|
def test_no_provider_returns_clear_error(self):
|
|
result = self._run({"prompt": "a dog"})
|
|
assert result["success"] is False
|
|
assert result["error_type"] == "no_provider_configured"
|
|
|
|
def test_unknown_provider_returns_clear_error(self):
|
|
result = self._run({"prompt": "a dog"}, configured="ghost")
|
|
assert result["success"] is False
|
|
assert result["error_type"] == "provider_not_registered"
|
|
|
|
def test_text_to_video_routes_without_image_url(self):
|
|
provider = _RecordingProvider("rec")
|
|
video_gen_registry.register_provider(provider)
|
|
result = self._run({"prompt": "a happy dog"})
|
|
assert result["success"] is True
|
|
assert result["modality"] == "text"
|
|
assert "image_url" not in provider.last_kwargs
|
|
assert provider.last_kwargs["aspect_ratio"] == "16:9"
|
|
assert provider.last_kwargs["resolution"] == "720p"
|
|
|
|
def test_image_to_video_routes_with_image_url(self):
|
|
provider = _RecordingProvider("rec")
|
|
video_gen_registry.register_provider(provider)
|
|
result = self._run({
|
|
"prompt": "animate this",
|
|
"image_url": "https://example.com/img.png",
|
|
})
|
|
assert result["success"] is True
|
|
assert result["modality"] == "image"
|
|
assert provider.last_kwargs["image_url"] == "https://example.com/img.png"
|
|
|
|
def test_prompt_required(self):
|
|
provider = _RecordingProvider("rec")
|
|
video_gen_registry.register_provider(provider)
|
|
result = self._run({"prompt": "", "image_url": "https://example.com/i.png"})
|
|
assert "error" in result
|
|
assert "prompt" in result["error"].lower()
|
|
|
|
def test_edit_extend_args_are_rejected_by_generate_tool(self):
|
|
provider = _RecordingProvider("rec")
|
|
video_gen_registry.register_provider(provider)
|
|
result = self._run({
|
|
"prompt": "make it rain",
|
|
"operation": "edit",
|
|
"video_url": "https://example.com/in.mp4",
|
|
})
|
|
assert "error" in result
|
|
assert "provider-specific tool" in result["error"]
|
|
|
|
def test_provider_exception_caught(self):
|
|
video_gen_registry.register_provider(_RaisingProvider())
|
|
result = self._run({"prompt": "x"})
|
|
assert result["success"] is False
|
|
assert result["error_type"] == "provider_exception"
|
|
|
|
def test_edit_extend_fields_not_in_schema(self):
|
|
from tools.video_generation_tool import VIDEO_GENERATE_SCHEMA
|
|
props = VIDEO_GENERATE_SCHEMA["parameters"]["properties"]
|
|
assert "operation" not in props
|
|
assert "video_url" not in props
|