mirror of
https://github.com/comfyanonymous/ComfyUI.git
synced 2025-09-11 20:17:30 +00:00
api_nodes: add MinimaxHailuoVideoNode node (#9262)
This commit is contained in:
13
comfy_api_nodes/apis/__init__.py
generated
13
comfy_api_nodes/apis/__init__.py
generated
@@ -1623,13 +1623,14 @@ class MinimaxTaskResultResponse(BaseModel):
|
|||||||
task_id: str = Field(..., description='The task ID being queried.')
|
task_id: str = Field(..., description='The task ID being queried.')
|
||||||
|
|
||||||
|
|
||||||
class Model(str, Enum):
|
class MiniMaxModel(str, Enum):
|
||||||
T2V_01_Director = 'T2V-01-Director'
|
T2V_01_Director = 'T2V-01-Director'
|
||||||
I2V_01_Director = 'I2V-01-Director'
|
I2V_01_Director = 'I2V-01-Director'
|
||||||
S2V_01 = 'S2V-01'
|
S2V_01 = 'S2V-01'
|
||||||
I2V_01 = 'I2V-01'
|
I2V_01 = 'I2V-01'
|
||||||
I2V_01_live = 'I2V-01-live'
|
I2V_01_live = 'I2V-01-live'
|
||||||
T2V_01 = 'T2V-01'
|
T2V_01 = 'T2V-01'
|
||||||
|
Hailuo_02 = 'MiniMax-Hailuo-02'
|
||||||
|
|
||||||
|
|
||||||
class SubjectReferenceItem(BaseModel):
|
class SubjectReferenceItem(BaseModel):
|
||||||
@@ -1651,7 +1652,7 @@ class MinimaxVideoGenerationRequest(BaseModel):
|
|||||||
None,
|
None,
|
||||||
description='URL or base64 encoding of the first frame image. Required when model is I2V-01, I2V-01-Director, or I2V-01-live.',
|
description='URL or base64 encoding of the first frame image. Required when model is I2V-01, I2V-01-Director, or I2V-01-live.',
|
||||||
)
|
)
|
||||||
model: Model = Field(
|
model: MiniMaxModel = Field(
|
||||||
...,
|
...,
|
||||||
description='Required. ID of model. Options: T2V-01-Director, I2V-01-Director, S2V-01, I2V-01, I2V-01-live, T2V-01',
|
description='Required. ID of model. Options: T2V-01-Director, I2V-01-Director, S2V-01, I2V-01, I2V-01-live, T2V-01',
|
||||||
)
|
)
|
||||||
@@ -1668,6 +1669,14 @@ class MinimaxVideoGenerationRequest(BaseModel):
|
|||||||
None,
|
None,
|
||||||
description='Only available when model is S2V-01. The model will generate a video based on the subject uploaded through this parameter.',
|
description='Only available when model is S2V-01. The model will generate a video based on the subject uploaded through this parameter.',
|
||||||
)
|
)
|
||||||
|
duration: Optional[int] = Field(
|
||||||
|
None,
|
||||||
|
description="The length of the output video in seconds."
|
||||||
|
)
|
||||||
|
resolution: Optional[str] = Field(
|
||||||
|
None,
|
||||||
|
description="The dimensions of the video display. 1080p corresponds to 1920 x 1080 pixels, 768p corresponds to 1366 x 768 pixels."
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
class MinimaxVideoGenerationResponse(BaseModel):
|
class MinimaxVideoGenerationResponse(BaseModel):
|
||||||
|
@@ -1,3 +1,4 @@
|
|||||||
|
from inspect import cleandoc
|
||||||
from typing import Union
|
from typing import Union
|
||||||
import logging
|
import logging
|
||||||
import torch
|
import torch
|
||||||
@@ -10,7 +11,7 @@ from comfy_api_nodes.apis import (
|
|||||||
MinimaxFileRetrieveResponse,
|
MinimaxFileRetrieveResponse,
|
||||||
MinimaxTaskResultResponse,
|
MinimaxTaskResultResponse,
|
||||||
SubjectReferenceItem,
|
SubjectReferenceItem,
|
||||||
Model
|
MiniMaxModel
|
||||||
)
|
)
|
||||||
from comfy_api_nodes.apis.client import (
|
from comfy_api_nodes.apis.client import (
|
||||||
ApiEndpoint,
|
ApiEndpoint,
|
||||||
@@ -84,7 +85,6 @@ class MinimaxTextToVideoNode:
|
|||||||
FUNCTION = "generate_video"
|
FUNCTION = "generate_video"
|
||||||
CATEGORY = "api node/video/MiniMax"
|
CATEGORY = "api node/video/MiniMax"
|
||||||
API_NODE = True
|
API_NODE = True
|
||||||
OUTPUT_NODE = True
|
|
||||||
|
|
||||||
async def generate_video(
|
async def generate_video(
|
||||||
self,
|
self,
|
||||||
@@ -121,7 +121,7 @@ class MinimaxTextToVideoNode:
|
|||||||
response_model=MinimaxVideoGenerationResponse,
|
response_model=MinimaxVideoGenerationResponse,
|
||||||
),
|
),
|
||||||
request=MinimaxVideoGenerationRequest(
|
request=MinimaxVideoGenerationRequest(
|
||||||
model=Model(model),
|
model=MiniMaxModel(model),
|
||||||
prompt=prompt_text,
|
prompt=prompt_text,
|
||||||
callback_url=None,
|
callback_url=None,
|
||||||
first_frame_image=image_url,
|
first_frame_image=image_url,
|
||||||
@@ -251,7 +251,6 @@ class MinimaxImageToVideoNode(MinimaxTextToVideoNode):
|
|||||||
FUNCTION = "generate_video"
|
FUNCTION = "generate_video"
|
||||||
CATEGORY = "api node/video/MiniMax"
|
CATEGORY = "api node/video/MiniMax"
|
||||||
API_NODE = True
|
API_NODE = True
|
||||||
OUTPUT_NODE = True
|
|
||||||
|
|
||||||
|
|
||||||
class MinimaxSubjectToVideoNode(MinimaxTextToVideoNode):
|
class MinimaxSubjectToVideoNode(MinimaxTextToVideoNode):
|
||||||
@@ -313,7 +312,181 @@ class MinimaxSubjectToVideoNode(MinimaxTextToVideoNode):
|
|||||||
FUNCTION = "generate_video"
|
FUNCTION = "generate_video"
|
||||||
CATEGORY = "api node/video/MiniMax"
|
CATEGORY = "api node/video/MiniMax"
|
||||||
API_NODE = True
|
API_NODE = True
|
||||||
OUTPUT_NODE = True
|
|
||||||
|
|
||||||
|
class MinimaxHailuoVideoNode:
|
||||||
|
"""Generates videos from prompt, with optional start frame using the new MiniMax Hailuo-02 model."""
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def INPUT_TYPES(s):
|
||||||
|
return {
|
||||||
|
"required": {
|
||||||
|
"prompt_text": (
|
||||||
|
"STRING",
|
||||||
|
{
|
||||||
|
"multiline": True,
|
||||||
|
"default": "",
|
||||||
|
"tooltip": "Text prompt to guide the video generation.",
|
||||||
|
},
|
||||||
|
),
|
||||||
|
},
|
||||||
|
"optional": {
|
||||||
|
"seed": (
|
||||||
|
IO.INT,
|
||||||
|
{
|
||||||
|
"default": 0,
|
||||||
|
"min": 0,
|
||||||
|
"max": 0xFFFFFFFFFFFFFFFF,
|
||||||
|
"control_after_generate": True,
|
||||||
|
"tooltip": "The random seed used for creating the noise.",
|
||||||
|
},
|
||||||
|
),
|
||||||
|
"first_frame_image": (
|
||||||
|
IO.IMAGE,
|
||||||
|
{
|
||||||
|
"tooltip": "Optional image to use as the first frame to generate a video."
|
||||||
|
},
|
||||||
|
),
|
||||||
|
"prompt_optimizer": (
|
||||||
|
IO.BOOLEAN,
|
||||||
|
{
|
||||||
|
"tooltip": "Optimize prompt to improve generation quality when needed.",
|
||||||
|
"default": True,
|
||||||
|
},
|
||||||
|
),
|
||||||
|
"duration": (
|
||||||
|
IO.COMBO,
|
||||||
|
{
|
||||||
|
"tooltip": "The length of the output video in seconds.",
|
||||||
|
"default": 6,
|
||||||
|
"options": [6, 10],
|
||||||
|
},
|
||||||
|
),
|
||||||
|
"resolution": (
|
||||||
|
IO.COMBO,
|
||||||
|
{
|
||||||
|
"tooltip": "The dimensions of the video display. "
|
||||||
|
"1080p corresponds to 1920 x 1080 pixels, 768p corresponds to 1366 x 768 pixels.",
|
||||||
|
"default": "768P",
|
||||||
|
"options": ["768P", "1080P"],
|
||||||
|
},
|
||||||
|
),
|
||||||
|
},
|
||||||
|
"hidden": {
|
||||||
|
"auth_token": "AUTH_TOKEN_COMFY_ORG",
|
||||||
|
"comfy_api_key": "API_KEY_COMFY_ORG",
|
||||||
|
"unique_id": "UNIQUE_ID",
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
RETURN_TYPES = ("VIDEO",)
|
||||||
|
DESCRIPTION = cleandoc(__doc__ or "")
|
||||||
|
FUNCTION = "generate_video"
|
||||||
|
CATEGORY = "api node/video/MiniMax"
|
||||||
|
API_NODE = True
|
||||||
|
|
||||||
|
async def generate_video(
|
||||||
|
self,
|
||||||
|
prompt_text,
|
||||||
|
seed=0,
|
||||||
|
first_frame_image: torch.Tensor=None, # used for ImageToVideo
|
||||||
|
prompt_optimizer=True,
|
||||||
|
duration=6,
|
||||||
|
resolution="768P",
|
||||||
|
model="MiniMax-Hailuo-02",
|
||||||
|
unique_id: Union[str, None]=None,
|
||||||
|
**kwargs,
|
||||||
|
):
|
||||||
|
if first_frame_image is None:
|
||||||
|
validate_string(prompt_text, field_name="prompt_text")
|
||||||
|
|
||||||
|
if model == "MiniMax-Hailuo-02" and resolution.upper() == "1080P" and duration != 6:
|
||||||
|
raise Exception(
|
||||||
|
"When model is MiniMax-Hailuo-02 and resolution is 1080P, duration is limited to 6 seconds."
|
||||||
|
)
|
||||||
|
|
||||||
|
# upload image, if passed in
|
||||||
|
image_url = None
|
||||||
|
if first_frame_image is not None:
|
||||||
|
image_url = (await upload_images_to_comfyapi(first_frame_image, max_images=1, auth_kwargs=kwargs))[0]
|
||||||
|
|
||||||
|
video_generate_operation = SynchronousOperation(
|
||||||
|
endpoint=ApiEndpoint(
|
||||||
|
path="/proxy/minimax/video_generation",
|
||||||
|
method=HttpMethod.POST,
|
||||||
|
request_model=MinimaxVideoGenerationRequest,
|
||||||
|
response_model=MinimaxVideoGenerationResponse,
|
||||||
|
),
|
||||||
|
request=MinimaxVideoGenerationRequest(
|
||||||
|
model=MiniMaxModel(model),
|
||||||
|
prompt=prompt_text,
|
||||||
|
callback_url=None,
|
||||||
|
first_frame_image=image_url,
|
||||||
|
prompt_optimizer=prompt_optimizer,
|
||||||
|
duration=duration,
|
||||||
|
resolution=resolution,
|
||||||
|
),
|
||||||
|
auth_kwargs=kwargs,
|
||||||
|
)
|
||||||
|
response = await video_generate_operation.execute()
|
||||||
|
|
||||||
|
task_id = response.task_id
|
||||||
|
if not task_id:
|
||||||
|
raise Exception(f"MiniMax generation failed: {response.base_resp}")
|
||||||
|
|
||||||
|
average_duration = 120 if resolution == "768P" else 240
|
||||||
|
video_generate_operation = PollingOperation(
|
||||||
|
poll_endpoint=ApiEndpoint(
|
||||||
|
path="/proxy/minimax/query/video_generation",
|
||||||
|
method=HttpMethod.GET,
|
||||||
|
request_model=EmptyRequest,
|
||||||
|
response_model=MinimaxTaskResultResponse,
|
||||||
|
query_params={"task_id": task_id},
|
||||||
|
),
|
||||||
|
completed_statuses=["Success"],
|
||||||
|
failed_statuses=["Fail"],
|
||||||
|
status_extractor=lambda x: x.status.value,
|
||||||
|
estimated_duration=average_duration,
|
||||||
|
node_id=unique_id,
|
||||||
|
auth_kwargs=kwargs,
|
||||||
|
)
|
||||||
|
task_result = await video_generate_operation.execute()
|
||||||
|
|
||||||
|
file_id = task_result.file_id
|
||||||
|
if file_id is None:
|
||||||
|
raise Exception("Request was not successful. Missing file ID.")
|
||||||
|
file_retrieve_operation = SynchronousOperation(
|
||||||
|
endpoint=ApiEndpoint(
|
||||||
|
path="/proxy/minimax/files/retrieve",
|
||||||
|
method=HttpMethod.GET,
|
||||||
|
request_model=EmptyRequest,
|
||||||
|
response_model=MinimaxFileRetrieveResponse,
|
||||||
|
query_params={"file_id": int(file_id)},
|
||||||
|
),
|
||||||
|
request=EmptyRequest(),
|
||||||
|
auth_kwargs=kwargs,
|
||||||
|
)
|
||||||
|
file_result = await file_retrieve_operation.execute()
|
||||||
|
|
||||||
|
file_url = file_result.file.download_url
|
||||||
|
if file_url is None:
|
||||||
|
raise Exception(
|
||||||
|
f"No video was found in the response. Full response: {file_result.model_dump()}"
|
||||||
|
)
|
||||||
|
logging.info(f"Generated video URL: {file_url}")
|
||||||
|
if unique_id:
|
||||||
|
if hasattr(file_result.file, "backup_download_url"):
|
||||||
|
message = f"Result URL: {file_url}\nBackup URL: {file_result.file.backup_download_url}"
|
||||||
|
else:
|
||||||
|
message = f"Result URL: {file_url}"
|
||||||
|
PromptServer.instance.send_progress_text(message, unique_id)
|
||||||
|
|
||||||
|
video_io = await download_url_to_bytesio(file_url)
|
||||||
|
if video_io is None:
|
||||||
|
error_msg = f"Failed to download video from {file_url}"
|
||||||
|
logging.error(error_msg)
|
||||||
|
raise Exception(error_msg)
|
||||||
|
return (VideoFromFile(video_io),)
|
||||||
|
|
||||||
|
|
||||||
# A dictionary that contains all nodes you want to export with their names
|
# A dictionary that contains all nodes you want to export with their names
|
||||||
@@ -322,6 +495,7 @@ NODE_CLASS_MAPPINGS = {
|
|||||||
"MinimaxTextToVideoNode": MinimaxTextToVideoNode,
|
"MinimaxTextToVideoNode": MinimaxTextToVideoNode,
|
||||||
"MinimaxImageToVideoNode": MinimaxImageToVideoNode,
|
"MinimaxImageToVideoNode": MinimaxImageToVideoNode,
|
||||||
# "MinimaxSubjectToVideoNode": MinimaxSubjectToVideoNode,
|
# "MinimaxSubjectToVideoNode": MinimaxSubjectToVideoNode,
|
||||||
|
"MinimaxHailuoVideoNode": MinimaxHailuoVideoNode,
|
||||||
}
|
}
|
||||||
|
|
||||||
# A dictionary that contains the friendly/humanly readable titles for the nodes
|
# A dictionary that contains the friendly/humanly readable titles for the nodes
|
||||||
@@ -329,4 +503,5 @@ NODE_DISPLAY_NAME_MAPPINGS = {
|
|||||||
"MinimaxTextToVideoNode": "MiniMax Text to Video",
|
"MinimaxTextToVideoNode": "MiniMax Text to Video",
|
||||||
"MinimaxImageToVideoNode": "MiniMax Image to Video",
|
"MinimaxImageToVideoNode": "MiniMax Image to Video",
|
||||||
"MinimaxSubjectToVideoNode": "MiniMax Subject to Video",
|
"MinimaxSubjectToVideoNode": "MiniMax Subject to Video",
|
||||||
|
"MinimaxHailuoVideoNode": "MiniMax Hailuo Video",
|
||||||
}
|
}
|
||||||
|
Reference in New Issue
Block a user