|
|
from inspect import cleandoc |
|
|
from typing import Optional |
|
|
from comfy_api_nodes.apis.pixverse_api import ( |
|
|
PixverseTextVideoRequest, |
|
|
PixverseImageVideoRequest, |
|
|
PixverseTransitionVideoRequest, |
|
|
PixverseImageUploadResponse, |
|
|
PixverseVideoResponse, |
|
|
PixverseGenerationStatusResponse, |
|
|
PixverseAspectRatio, |
|
|
PixverseQuality, |
|
|
PixverseDuration, |
|
|
PixverseMotionMode, |
|
|
PixverseStatus, |
|
|
PixverseIO, |
|
|
pixverse_templates, |
|
|
) |
|
|
from comfy_api_nodes.apis.client import ( |
|
|
ApiEndpoint, |
|
|
HttpMethod, |
|
|
SynchronousOperation, |
|
|
PollingOperation, |
|
|
EmptyRequest, |
|
|
) |
|
|
from comfy_api_nodes.apinode_utils import ( |
|
|
tensor_to_bytesio, |
|
|
validate_string, |
|
|
) |
|
|
from comfy.comfy_types.node_typing import IO, ComfyNodeABC |
|
|
from comfy_api.input_impl import VideoFromFile |
|
|
|
|
|
import torch |
|
|
import aiohttp |
|
|
from io import BytesIO |
|
|
|
|
|
|
|
|
AVERAGE_DURATION_T2V = 32 |
|
|
AVERAGE_DURATION_I2V = 30 |
|
|
AVERAGE_DURATION_T2T = 52 |
|
|
|
|
|
|
|
|
def get_video_url_from_response( |
|
|
response: PixverseGenerationStatusResponse, |
|
|
) -> Optional[str]: |
|
|
if response.Resp is None or response.Resp.url is None: |
|
|
return None |
|
|
return str(response.Resp.url) |
|
|
|
|
|
|
|
|
async def upload_image_to_pixverse(image: torch.Tensor, auth_kwargs=None): |
|
|
|
|
|
files = {"image": tensor_to_bytesio(image)} |
|
|
operation = SynchronousOperation( |
|
|
endpoint=ApiEndpoint( |
|
|
path="/proxy/pixverse/image/upload", |
|
|
method=HttpMethod.POST, |
|
|
request_model=EmptyRequest, |
|
|
response_model=PixverseImageUploadResponse, |
|
|
), |
|
|
request=EmptyRequest(), |
|
|
files=files, |
|
|
content_type="multipart/form-data", |
|
|
auth_kwargs=auth_kwargs, |
|
|
) |
|
|
response_upload: PixverseImageUploadResponse = await operation.execute() |
|
|
|
|
|
if response_upload.Resp is None: |
|
|
raise Exception( |
|
|
f"PixVerse image upload request failed: '{response_upload.ErrMsg}'" |
|
|
) |
|
|
|
|
|
return response_upload.Resp.img_id |
|
|
|
|
|
|
|
|
class PixverseTemplateNode: |
|
|
""" |
|
|
Select template for PixVerse Video generation. |
|
|
""" |
|
|
|
|
|
RETURN_TYPES = (PixverseIO.TEMPLATE,) |
|
|
RETURN_NAMES = ("pixverse_template",) |
|
|
FUNCTION = "create_template" |
|
|
CATEGORY = "api node/video/PixVerse" |
|
|
|
|
|
@classmethod |
|
|
def INPUT_TYPES(s): |
|
|
return { |
|
|
"required": { |
|
|
"template": (list(pixverse_templates.keys()),), |
|
|
} |
|
|
} |
|
|
|
|
|
def create_template(self, template: str): |
|
|
template_id = pixverse_templates.get(template, None) |
|
|
if template_id is None: |
|
|
raise Exception(f"Template '{template}' is not recognized.") |
|
|
|
|
|
return (template_id,) |
|
|
|
|
|
|
|
|
class PixverseTextToVideoNode(ComfyNodeABC): |
|
|
""" |
|
|
Generates videos based on prompt and output_size. |
|
|
""" |
|
|
|
|
|
RETURN_TYPES = (IO.VIDEO,) |
|
|
DESCRIPTION = cleandoc(__doc__ or "") |
|
|
FUNCTION = "api_call" |
|
|
API_NODE = True |
|
|
CATEGORY = "api node/video/PixVerse" |
|
|
|
|
|
@classmethod |
|
|
def INPUT_TYPES(s): |
|
|
return { |
|
|
"required": { |
|
|
"prompt": ( |
|
|
IO.STRING, |
|
|
{ |
|
|
"multiline": True, |
|
|
"default": "", |
|
|
"tooltip": "Prompt for the video generation", |
|
|
}, |
|
|
), |
|
|
"aspect_ratio": ([ratio.value for ratio in PixverseAspectRatio],), |
|
|
"quality": ( |
|
|
[resolution.value for resolution in PixverseQuality], |
|
|
{ |
|
|
"default": PixverseQuality.res_540p, |
|
|
}, |
|
|
), |
|
|
"duration_seconds": ([dur.value for dur in PixverseDuration],), |
|
|
"motion_mode": ([mode.value for mode in PixverseMotionMode],), |
|
|
"seed": ( |
|
|
IO.INT, |
|
|
{ |
|
|
"default": 0, |
|
|
"min": 0, |
|
|
"max": 2147483647, |
|
|
"control_after_generate": True, |
|
|
"tooltip": "Seed for video generation.", |
|
|
}, |
|
|
), |
|
|
}, |
|
|
"optional": { |
|
|
"negative_prompt": ( |
|
|
IO.STRING, |
|
|
{ |
|
|
"default": "", |
|
|
"forceInput": True, |
|
|
"tooltip": "An optional text description of undesired elements on an image.", |
|
|
}, |
|
|
), |
|
|
"pixverse_template": ( |
|
|
PixverseIO.TEMPLATE, |
|
|
{ |
|
|
"tooltip": "An optional template to influence style of generation, created by the PixVerse Template node." |
|
|
}, |
|
|
), |
|
|
}, |
|
|
"hidden": { |
|
|
"auth_token": "AUTH_TOKEN_COMFY_ORG", |
|
|
"comfy_api_key": "API_KEY_COMFY_ORG", |
|
|
"unique_id": "UNIQUE_ID", |
|
|
}, |
|
|
} |
|
|
|
|
|
async def api_call( |
|
|
self, |
|
|
prompt: str, |
|
|
aspect_ratio: str, |
|
|
quality: str, |
|
|
duration_seconds: int, |
|
|
motion_mode: str, |
|
|
seed, |
|
|
negative_prompt: str = None, |
|
|
pixverse_template: int = None, |
|
|
unique_id: Optional[str] = None, |
|
|
**kwargs, |
|
|
): |
|
|
validate_string(prompt, strip_whitespace=False) |
|
|
|
|
|
|
|
|
if quality == PixverseQuality.res_1080p: |
|
|
motion_mode = PixverseMotionMode.normal |
|
|
duration_seconds = PixverseDuration.dur_5 |
|
|
elif duration_seconds != PixverseDuration.dur_5: |
|
|
motion_mode = PixverseMotionMode.normal |
|
|
|
|
|
operation = SynchronousOperation( |
|
|
endpoint=ApiEndpoint( |
|
|
path="/proxy/pixverse/video/text/generate", |
|
|
method=HttpMethod.POST, |
|
|
request_model=PixverseTextVideoRequest, |
|
|
response_model=PixverseVideoResponse, |
|
|
), |
|
|
request=PixverseTextVideoRequest( |
|
|
prompt=prompt, |
|
|
aspect_ratio=aspect_ratio, |
|
|
quality=quality, |
|
|
duration=duration_seconds, |
|
|
motion_mode=motion_mode, |
|
|
negative_prompt=negative_prompt if negative_prompt else None, |
|
|
template_id=pixverse_template, |
|
|
seed=seed, |
|
|
), |
|
|
auth_kwargs=kwargs, |
|
|
) |
|
|
response_api = await operation.execute() |
|
|
|
|
|
if response_api.Resp is None: |
|
|
raise Exception(f"PixVerse request failed: '{response_api.ErrMsg}'") |
|
|
|
|
|
operation = PollingOperation( |
|
|
poll_endpoint=ApiEndpoint( |
|
|
path=f"/proxy/pixverse/video/result/{response_api.Resp.video_id}", |
|
|
method=HttpMethod.GET, |
|
|
request_model=EmptyRequest, |
|
|
response_model=PixverseGenerationStatusResponse, |
|
|
), |
|
|
completed_statuses=[PixverseStatus.successful], |
|
|
failed_statuses=[ |
|
|
PixverseStatus.contents_moderation, |
|
|
PixverseStatus.failed, |
|
|
PixverseStatus.deleted, |
|
|
], |
|
|
status_extractor=lambda x: x.Resp.status, |
|
|
auth_kwargs=kwargs, |
|
|
node_id=unique_id, |
|
|
result_url_extractor=get_video_url_from_response, |
|
|
estimated_duration=AVERAGE_DURATION_T2V, |
|
|
) |
|
|
response_poll = await operation.execute() |
|
|
|
|
|
async with aiohttp.ClientSession() as session: |
|
|
async with session.get(response_poll.Resp.url) as vid_response: |
|
|
return (VideoFromFile(BytesIO(await vid_response.content.read())),) |
|
|
|
|
|
|
|
|
class PixverseImageToVideoNode(ComfyNodeABC): |
|
|
""" |
|
|
Generates videos based on prompt and output_size. |
|
|
""" |
|
|
|
|
|
RETURN_TYPES = (IO.VIDEO,) |
|
|
DESCRIPTION = cleandoc(__doc__ or "") |
|
|
FUNCTION = "api_call" |
|
|
API_NODE = True |
|
|
CATEGORY = "api node/video/PixVerse" |
|
|
|
|
|
@classmethod |
|
|
def INPUT_TYPES(s): |
|
|
return { |
|
|
"required": { |
|
|
"image": (IO.IMAGE,), |
|
|
"prompt": ( |
|
|
IO.STRING, |
|
|
{ |
|
|
"multiline": True, |
|
|
"default": "", |
|
|
"tooltip": "Prompt for the video generation", |
|
|
}, |
|
|
), |
|
|
"quality": ( |
|
|
[resolution.value for resolution in PixverseQuality], |
|
|
{ |
|
|
"default": PixverseQuality.res_540p, |
|
|
}, |
|
|
), |
|
|
"duration_seconds": ([dur.value for dur in PixverseDuration],), |
|
|
"motion_mode": ([mode.value for mode in PixverseMotionMode],), |
|
|
"seed": ( |
|
|
IO.INT, |
|
|
{ |
|
|
"default": 0, |
|
|
"min": 0, |
|
|
"max": 2147483647, |
|
|
"control_after_generate": True, |
|
|
"tooltip": "Seed for video generation.", |
|
|
}, |
|
|
), |
|
|
}, |
|
|
"optional": { |
|
|
"negative_prompt": ( |
|
|
IO.STRING, |
|
|
{ |
|
|
"default": "", |
|
|
"forceInput": True, |
|
|
"tooltip": "An optional text description of undesired elements on an image.", |
|
|
}, |
|
|
), |
|
|
"pixverse_template": ( |
|
|
PixverseIO.TEMPLATE, |
|
|
{ |
|
|
"tooltip": "An optional template to influence style of generation, created by the PixVerse Template node." |
|
|
}, |
|
|
), |
|
|
}, |
|
|
"hidden": { |
|
|
"auth_token": "AUTH_TOKEN_COMFY_ORG", |
|
|
"comfy_api_key": "API_KEY_COMFY_ORG", |
|
|
"unique_id": "UNIQUE_ID", |
|
|
}, |
|
|
} |
|
|
|
|
|
async def api_call( |
|
|
self, |
|
|
image: torch.Tensor, |
|
|
prompt: str, |
|
|
quality: str, |
|
|
duration_seconds: int, |
|
|
motion_mode: str, |
|
|
seed, |
|
|
negative_prompt: str = None, |
|
|
pixverse_template: int = None, |
|
|
unique_id: Optional[str] = None, |
|
|
**kwargs, |
|
|
): |
|
|
validate_string(prompt, strip_whitespace=False) |
|
|
img_id = await upload_image_to_pixverse(image, auth_kwargs=kwargs) |
|
|
|
|
|
|
|
|
|
|
|
if quality == PixverseQuality.res_1080p: |
|
|
motion_mode = PixverseMotionMode.normal |
|
|
duration_seconds = PixverseDuration.dur_5 |
|
|
elif duration_seconds != PixverseDuration.dur_5: |
|
|
motion_mode = PixverseMotionMode.normal |
|
|
|
|
|
operation = SynchronousOperation( |
|
|
endpoint=ApiEndpoint( |
|
|
path="/proxy/pixverse/video/img/generate", |
|
|
method=HttpMethod.POST, |
|
|
request_model=PixverseImageVideoRequest, |
|
|
response_model=PixverseVideoResponse, |
|
|
), |
|
|
request=PixverseImageVideoRequest( |
|
|
img_id=img_id, |
|
|
prompt=prompt, |
|
|
quality=quality, |
|
|
duration=duration_seconds, |
|
|
motion_mode=motion_mode, |
|
|
negative_prompt=negative_prompt if negative_prompt else None, |
|
|
template_id=pixverse_template, |
|
|
seed=seed, |
|
|
), |
|
|
auth_kwargs=kwargs, |
|
|
) |
|
|
response_api = await operation.execute() |
|
|
|
|
|
if response_api.Resp is None: |
|
|
raise Exception(f"PixVerse request failed: '{response_api.ErrMsg}'") |
|
|
|
|
|
operation = PollingOperation( |
|
|
poll_endpoint=ApiEndpoint( |
|
|
path=f"/proxy/pixverse/video/result/{response_api.Resp.video_id}", |
|
|
method=HttpMethod.GET, |
|
|
request_model=EmptyRequest, |
|
|
response_model=PixverseGenerationStatusResponse, |
|
|
), |
|
|
completed_statuses=[PixverseStatus.successful], |
|
|
failed_statuses=[ |
|
|
PixverseStatus.contents_moderation, |
|
|
PixverseStatus.failed, |
|
|
PixverseStatus.deleted, |
|
|
], |
|
|
status_extractor=lambda x: x.Resp.status, |
|
|
auth_kwargs=kwargs, |
|
|
node_id=unique_id, |
|
|
result_url_extractor=get_video_url_from_response, |
|
|
estimated_duration=AVERAGE_DURATION_I2V, |
|
|
) |
|
|
response_poll = await operation.execute() |
|
|
|
|
|
async with aiohttp.ClientSession() as session: |
|
|
async with session.get(response_poll.Resp.url) as vid_response: |
|
|
return (VideoFromFile(BytesIO(await vid_response.content.read())),) |
|
|
|
|
|
|
|
|
class PixverseTransitionVideoNode(ComfyNodeABC): |
|
|
""" |
|
|
Generates videos based on prompt and output_size. |
|
|
""" |
|
|
|
|
|
RETURN_TYPES = (IO.VIDEO,) |
|
|
DESCRIPTION = cleandoc(__doc__ or "") |
|
|
FUNCTION = "api_call" |
|
|
API_NODE = True |
|
|
CATEGORY = "api node/video/PixVerse" |
|
|
|
|
|
@classmethod |
|
|
def INPUT_TYPES(s): |
|
|
return { |
|
|
"required": { |
|
|
"first_frame": (IO.IMAGE,), |
|
|
"last_frame": (IO.IMAGE,), |
|
|
"prompt": ( |
|
|
IO.STRING, |
|
|
{ |
|
|
"multiline": True, |
|
|
"default": "", |
|
|
"tooltip": "Prompt for the video generation", |
|
|
}, |
|
|
), |
|
|
"quality": ( |
|
|
[resolution.value for resolution in PixverseQuality], |
|
|
{ |
|
|
"default": PixverseQuality.res_540p, |
|
|
}, |
|
|
), |
|
|
"duration_seconds": ([dur.value for dur in PixverseDuration],), |
|
|
"motion_mode": ([mode.value for mode in PixverseMotionMode],), |
|
|
"seed": ( |
|
|
IO.INT, |
|
|
{ |
|
|
"default": 0, |
|
|
"min": 0, |
|
|
"max": 2147483647, |
|
|
"control_after_generate": True, |
|
|
"tooltip": "Seed for video generation.", |
|
|
}, |
|
|
), |
|
|
}, |
|
|
"optional": { |
|
|
"negative_prompt": ( |
|
|
IO.STRING, |
|
|
{ |
|
|
"default": "", |
|
|
"forceInput": True, |
|
|
"tooltip": "An optional text description of undesired elements on an image.", |
|
|
}, |
|
|
), |
|
|
}, |
|
|
"hidden": { |
|
|
"auth_token": "AUTH_TOKEN_COMFY_ORG", |
|
|
"comfy_api_key": "API_KEY_COMFY_ORG", |
|
|
"unique_id": "UNIQUE_ID", |
|
|
}, |
|
|
} |
|
|
|
|
|
async def api_call( |
|
|
self, |
|
|
first_frame: torch.Tensor, |
|
|
last_frame: torch.Tensor, |
|
|
prompt: str, |
|
|
quality: str, |
|
|
duration_seconds: int, |
|
|
motion_mode: str, |
|
|
seed, |
|
|
negative_prompt: str = None, |
|
|
unique_id: Optional[str] = None, |
|
|
**kwargs, |
|
|
): |
|
|
validate_string(prompt, strip_whitespace=False) |
|
|
first_frame_id = await upload_image_to_pixverse(first_frame, auth_kwargs=kwargs) |
|
|
last_frame_id = await upload_image_to_pixverse(last_frame, auth_kwargs=kwargs) |
|
|
|
|
|
|
|
|
|
|
|
if quality == PixverseQuality.res_1080p: |
|
|
motion_mode = PixverseMotionMode.normal |
|
|
duration_seconds = PixverseDuration.dur_5 |
|
|
elif duration_seconds != PixverseDuration.dur_5: |
|
|
motion_mode = PixverseMotionMode.normal |
|
|
|
|
|
operation = SynchronousOperation( |
|
|
endpoint=ApiEndpoint( |
|
|
path="/proxy/pixverse/video/transition/generate", |
|
|
method=HttpMethod.POST, |
|
|
request_model=PixverseTransitionVideoRequest, |
|
|
response_model=PixverseVideoResponse, |
|
|
), |
|
|
request=PixverseTransitionVideoRequest( |
|
|
first_frame_img=first_frame_id, |
|
|
last_frame_img=last_frame_id, |
|
|
prompt=prompt, |
|
|
quality=quality, |
|
|
duration=duration_seconds, |
|
|
motion_mode=motion_mode, |
|
|
negative_prompt=negative_prompt if negative_prompt else None, |
|
|
seed=seed, |
|
|
), |
|
|
auth_kwargs=kwargs, |
|
|
) |
|
|
response_api = await operation.execute() |
|
|
|
|
|
if response_api.Resp is None: |
|
|
raise Exception(f"PixVerse request failed: '{response_api.ErrMsg}'") |
|
|
|
|
|
operation = PollingOperation( |
|
|
poll_endpoint=ApiEndpoint( |
|
|
path=f"/proxy/pixverse/video/result/{response_api.Resp.video_id}", |
|
|
method=HttpMethod.GET, |
|
|
request_model=EmptyRequest, |
|
|
response_model=PixverseGenerationStatusResponse, |
|
|
), |
|
|
completed_statuses=[PixverseStatus.successful], |
|
|
failed_statuses=[ |
|
|
PixverseStatus.contents_moderation, |
|
|
PixverseStatus.failed, |
|
|
PixverseStatus.deleted, |
|
|
], |
|
|
status_extractor=lambda x: x.Resp.status, |
|
|
auth_kwargs=kwargs, |
|
|
node_id=unique_id, |
|
|
result_url_extractor=get_video_url_from_response, |
|
|
estimated_duration=AVERAGE_DURATION_T2V, |
|
|
) |
|
|
response_poll = await operation.execute() |
|
|
|
|
|
async with aiohttp.ClientSession() as session: |
|
|
async with session.get(response_poll.Resp.url) as vid_response: |
|
|
return (VideoFromFile(BytesIO(await vid_response.content.read())),) |
|
|
|
|
|
|
|
|
NODE_CLASS_MAPPINGS = { |
|
|
"PixverseTextToVideoNode": PixverseTextToVideoNode, |
|
|
"PixverseImageToVideoNode": PixverseImageToVideoNode, |
|
|
"PixverseTransitionVideoNode": PixverseTransitionVideoNode, |
|
|
"PixverseTemplateNode": PixverseTemplateNode, |
|
|
} |
|
|
|
|
|
NODE_DISPLAY_NAME_MAPPINGS = { |
|
|
"PixverseTextToVideoNode": "PixVerse Text to Video", |
|
|
"PixverseImageToVideoNode": "PixVerse Image to Video", |
|
|
"PixverseTransitionVideoNode": "PixVerse Transition Video", |
|
|
"PixverseTemplateNode": "PixVerse Template", |
|
|
} |
|
|
|