| from __future__ import annotations
|
| from inspect import cleandoc
|
| from typing import Optional
|
| from comfy.utils import ProgressBar
|
| from comfy_extras.nodes_images import SVG
|
| from comfy.comfy_types.node_typing import IO
|
| from comfy_api_nodes.apis.recraft_api import (
|
| RecraftImageGenerationRequest,
|
| RecraftImageGenerationResponse,
|
| RecraftImageSize,
|
| RecraftModel,
|
| RecraftStyle,
|
| RecraftStyleV3,
|
| RecraftColor,
|
| RecraftColorChain,
|
| RecraftControls,
|
| RecraftIO,
|
| get_v3_substyles,
|
| )
|
| from comfy_api_nodes.apis.client import (
|
| ApiEndpoint,
|
| HttpMethod,
|
| SynchronousOperation,
|
| EmptyRequest,
|
| )
|
| from comfy_api_nodes.apinode_utils import (
|
| bytesio_to_image_tensor,
|
| download_url_to_bytesio,
|
| tensor_to_bytesio,
|
| resize_mask_to_image,
|
| validate_string,
|
| )
|
| from server import PromptServer
|
|
|
| import torch
|
| from io import BytesIO
|
| from PIL import UnidentifiedImageError
|
|
|
|
|
| def handle_recraft_file_request(
|
| image: torch.Tensor,
|
| path: str,
|
| mask: torch.Tensor=None,
|
| total_pixels=4096*4096,
|
| timeout=1024,
|
| request=None,
|
| auth_kwargs: dict[str,str] = None,
|
| ) -> list[BytesIO]:
|
| """
|
| Handle sending common Recraft file-only request to get back file bytes.
|
| """
|
| if request is None:
|
| request = EmptyRequest()
|
|
|
| files = {
|
| 'image': tensor_to_bytesio(image, total_pixels=total_pixels).read()
|
| }
|
| if mask is not None:
|
| files['mask'] = tensor_to_bytesio(mask, total_pixels=total_pixels).read()
|
|
|
| operation = SynchronousOperation(
|
| endpoint=ApiEndpoint(
|
| path=path,
|
| method=HttpMethod.POST,
|
| request_model=type(request),
|
| response_model=RecraftImageGenerationResponse,
|
| ),
|
| request=request,
|
| files=files,
|
| content_type="multipart/form-data",
|
| auth_kwargs=auth_kwargs,
|
| multipart_parser=recraft_multipart_parser,
|
| )
|
| response: RecraftImageGenerationResponse = operation.execute()
|
| all_bytesio = []
|
| if response.image is not None:
|
| all_bytesio.append(download_url_to_bytesio(response.image.url, timeout=timeout))
|
| else:
|
| for data in response.data:
|
| all_bytesio.append(download_url_to_bytesio(data.url, timeout=timeout))
|
|
|
| return all_bytesio
|
|
|
|
|
| def recraft_multipart_parser(data, parent_key=None, formatter: callable=None, converted_to_check: list[list]=None, is_list=False) -> dict:
|
| """
|
| Formats data such that multipart/form-data will work with requests library
|
| when both files and data are present.
|
|
|
| The OpenAI client that Recraft uses has a bizarre way of serializing lists:
|
|
|
| It does NOT keep track of indeces of each list, so for background_color, that must be serialized as:
|
| 'background_color[rgb][]' = [0, 0, 255]
|
| where the array is assigned to a key that has '[]' at the end, to signal it's an array.
|
|
|
| This has the consequence of nested lists having the exact same key, forcing arrays to merge; all colors inputs fall under the same key:
|
| if 1 color -> 'controls[colors][][rgb][]' = [0, 0, 255]
|
| if 2 colors -> 'controls[colors][][rgb][]' = [0, 0, 255, 255, 0, 0]
|
| if 3 colors -> 'controls[colors][][rgb][]' = [0, 0, 255, 255, 0, 0, 0, 255, 0]
|
| etc.
|
| Whoever made this serialization up at OpenAI added the constraint that lists must be of uniform length on objects of same 'type'.
|
| """
|
|
|
|
|
|
|
| def handle_converted_lists(data, parent_key, lists_to_check=tuple[list]):
|
|
|
| for check_list in lists_to_check:
|
| for conv_tuple in check_list:
|
| if conv_tuple[0] == parent_key and type(conv_tuple[1]) is list:
|
| conv_tuple[1].append(formatter(data))
|
| return True
|
| return False
|
|
|
| if converted_to_check is None:
|
| converted_to_check = []
|
|
|
|
|
| if formatter is None:
|
| formatter = lambda v: v
|
|
|
| if type(data) is not dict:
|
|
|
| added = handle_converted_lists(data, parent_key, converted_to_check)
|
| if added:
|
| return {}
|
|
|
| if is_list:
|
| return {parent_key: [formatter(data)]}
|
|
|
| return {parent_key: formatter(data)}
|
|
|
| converted = []
|
| next_check = [converted]
|
| next_check.extend(converted_to_check)
|
|
|
| for key, value in data.items():
|
| current_key = key if parent_key is None else f"{parent_key}[{key}]"
|
| if type(value) is dict:
|
| converted.extend(recraft_multipart_parser(value, current_key, formatter, next_check).items())
|
| elif type(value) is list:
|
| for ind, list_value in enumerate(value):
|
| iter_key = f"{current_key}[]"
|
| converted.extend(recraft_multipart_parser(list_value, iter_key, formatter, next_check, is_list=True).items())
|
| else:
|
| converted.append((current_key, formatter(value)))
|
|
|
| return dict(converted)
|
|
|
|
|
| class handle_recraft_image_output:
|
| """
|
| Catch an exception related to receiving SVG data instead of image, when Infinite Style Library style_id is in use.
|
| """
|
| def __init__(self):
|
| pass
|
|
|
| def __enter__(self):
|
| pass
|
|
|
| def __exit__(self, exc_type, exc_val, exc_tb):
|
| if exc_type is not None and exc_type is UnidentifiedImageError:
|
| raise Exception("Received output data was not an image; likely an SVG. If you used style_id, make sure it is not a Vector art style.")
|
|
|
|
|
| class RecraftColorRGBNode:
|
| """
|
| Create Recraft Color by choosing specific RGB values.
|
| """
|
|
|
| RETURN_TYPES = (RecraftIO.COLOR,)
|
| DESCRIPTION = cleandoc(__doc__ or "")
|
| RETURN_NAMES = ("recraft_color",)
|
| FUNCTION = "create_color"
|
| CATEGORY = "api node/image/Recraft"
|
|
|
| @classmethod
|
| def INPUT_TYPES(s):
|
| return {
|
| "required": {
|
| "r": (IO.INT, {
|
| "default": 0,
|
| "min": 0,
|
| "max": 255,
|
| "tooltip": "Red value of color."
|
| }),
|
| "g": (IO.INT, {
|
| "default": 0,
|
| "min": 0,
|
| "max": 255,
|
| "tooltip": "Green value of color."
|
| }),
|
| "b": (IO.INT, {
|
| "default": 0,
|
| "min": 0,
|
| "max": 255,
|
| "tooltip": "Blue value of color."
|
| }),
|
| },
|
| "optional": {
|
| "recraft_color": (RecraftIO.COLOR,),
|
| }
|
| }
|
|
|
| def create_color(self, r: int, g: int, b: int, recraft_color: RecraftColorChain=None):
|
| recraft_color = recraft_color.clone() if recraft_color else RecraftColorChain()
|
| recraft_color.add(RecraftColor(r, g, b))
|
| return (recraft_color, )
|
|
|
|
|
| class RecraftControlsNode:
|
| """
|
| Create Recraft Controls for customizing Recraft generation.
|
| """
|
|
|
| RETURN_TYPES = (RecraftIO.CONTROLS,)
|
| RETURN_NAMES = ("recraft_controls",)
|
| DESCRIPTION = cleandoc(__doc__ or "")
|
| FUNCTION = "create_controls"
|
| CATEGORY = "api node/image/Recraft"
|
|
|
| @classmethod
|
| def INPUT_TYPES(s):
|
| return {
|
| "required": {
|
| },
|
| "optional": {
|
| "colors": (RecraftIO.COLOR,),
|
| "background_color": (RecraftIO.COLOR,),
|
| }
|
| }
|
|
|
| def create_controls(self, colors: RecraftColorChain=None, background_color: RecraftColorChain=None):
|
| return (RecraftControls(colors=colors, background_color=background_color), )
|
|
|
|
|
| class RecraftStyleV3RealisticImageNode:
|
| """
|
| Select realistic_image style and optional substyle.
|
| """
|
|
|
| RETURN_TYPES = (RecraftIO.STYLEV3,)
|
| RETURN_NAMES = ("recraft_style",)
|
| DESCRIPTION = cleandoc(__doc__ or "")
|
| FUNCTION = "create_style"
|
| CATEGORY = "api node/image/Recraft"
|
|
|
| RECRAFT_STYLE = RecraftStyleV3.realistic_image
|
|
|
| @classmethod
|
| def INPUT_TYPES(s):
|
| return {
|
| "required": {
|
| "substyle": (get_v3_substyles(s.RECRAFT_STYLE),),
|
| }
|
| }
|
|
|
| def create_style(self, substyle: str):
|
| if substyle == "None":
|
| substyle = None
|
| return (RecraftStyle(self.RECRAFT_STYLE, substyle),)
|
|
|
|
|
| class RecraftStyleV3DigitalIllustrationNode(RecraftStyleV3RealisticImageNode):
|
| """
|
| Select digital_illustration style and optional substyle.
|
| """
|
|
|
| RECRAFT_STYLE = RecraftStyleV3.digital_illustration
|
|
|
|
|
| class RecraftStyleV3VectorIllustrationNode(RecraftStyleV3RealisticImageNode):
|
| """
|
| Select vector_illustration style and optional substyle.
|
| """
|
|
|
| RECRAFT_STYLE = RecraftStyleV3.vector_illustration
|
|
|
|
|
| class RecraftStyleV3LogoRasterNode(RecraftStyleV3RealisticImageNode):
|
| """
|
| Select vector_illustration style and optional substyle.
|
| """
|
|
|
| @classmethod
|
| def INPUT_TYPES(s):
|
| return {
|
| "required": {
|
| "substyle": (get_v3_substyles(s.RECRAFT_STYLE, include_none=False),),
|
| }
|
| }
|
|
|
| RECRAFT_STYLE = RecraftStyleV3.logo_raster
|
|
|
|
|
| class RecraftStyleInfiniteStyleLibrary:
|
| """
|
| Select style based on preexisting UUID from Recraft's Infinite Style Library.
|
| """
|
|
|
| RETURN_TYPES = (RecraftIO.STYLEV3,)
|
| RETURN_NAMES = ("recraft_style",)
|
| DESCRIPTION = cleandoc(__doc__ or "")
|
| FUNCTION = "create_style"
|
| CATEGORY = "api node/image/Recraft"
|
|
|
| @classmethod
|
| def INPUT_TYPES(s):
|
| return {
|
| "required": {
|
| "style_id": (IO.STRING, {
|
| "default": "",
|
| "tooltip": "UUID of style from Infinite Style Library.",
|
| })
|
| }
|
| }
|
|
|
| def create_style(self, style_id: str):
|
| if not style_id:
|
| raise Exception("The style_id input cannot be empty.")
|
| return (RecraftStyle(style_id=style_id),)
|
|
|
|
|
| class RecraftTextToImageNode:
|
| """
|
| Generates images synchronously based on prompt and resolution.
|
| """
|
|
|
| RETURN_TYPES = (IO.IMAGE,)
|
| DESCRIPTION = cleandoc(__doc__ or "")
|
| FUNCTION = "api_call"
|
| API_NODE = True
|
| CATEGORY = "api node/image/Recraft"
|
|
|
| @classmethod
|
| def INPUT_TYPES(s):
|
| return {
|
| "required": {
|
| "prompt": (
|
| IO.STRING,
|
| {
|
| "multiline": True,
|
| "default": "",
|
| "tooltip": "Prompt for the image generation.",
|
| },
|
| ),
|
| "size": (
|
| [res.value for res in RecraftImageSize],
|
| {
|
| "default": RecraftImageSize.res_1024x1024,
|
| "tooltip": "The size of the generated image.",
|
| },
|
| ),
|
| "n": (
|
| IO.INT,
|
| {
|
| "default": 1,
|
| "min": 1,
|
| "max": 6,
|
| "tooltip": "The number of images to generate.",
|
| },
|
| ),
|
| "seed": (
|
| IO.INT,
|
| {
|
| "default": 0,
|
| "min": 0,
|
| "max": 0xFFFFFFFFFFFFFFFF,
|
| "control_after_generate": True,
|
| "tooltip": "Seed to determine if node should re-run; actual results are nondeterministic regardless of seed.",
|
| },
|
| ),
|
| },
|
| "optional": {
|
| "recraft_style": (RecraftIO.STYLEV3,),
|
| "negative_prompt": (
|
| IO.STRING,
|
| {
|
| "default": "",
|
| "forceInput": True,
|
| "tooltip": "An optional text description of undesired elements on an image.",
|
| },
|
| ),
|
| "recraft_controls": (
|
| RecraftIO.CONTROLS,
|
| {
|
| "tooltip": "Optional additional controls over the generation via the Recraft Controls node."
|
| },
|
| ),
|
| },
|
| "hidden": {
|
| "auth_token": "AUTH_TOKEN_COMFY_ORG",
|
| "comfy_api_key": "API_KEY_COMFY_ORG",
|
| "unique_id": "UNIQUE_ID",
|
| },
|
| }
|
|
|
| def api_call(
|
| self,
|
| prompt: str,
|
| size: str,
|
| n: int,
|
| seed,
|
| recraft_style: RecraftStyle = None,
|
| negative_prompt: str = None,
|
| recraft_controls: RecraftControls = None,
|
| unique_id: Optional[str] = None,
|
| **kwargs,
|
| ):
|
| validate_string(prompt, strip_whitespace=False, max_length=1000)
|
| default_style = RecraftStyle(RecraftStyleV3.realistic_image)
|
| if recraft_style is None:
|
| recraft_style = default_style
|
|
|
| controls_api = None
|
| if recraft_controls:
|
| controls_api = recraft_controls.create_api_model()
|
|
|
| if not negative_prompt:
|
| negative_prompt = None
|
|
|
| operation = SynchronousOperation(
|
| endpoint=ApiEndpoint(
|
| path="/proxy/recraft/image_generation",
|
| method=HttpMethod.POST,
|
| request_model=RecraftImageGenerationRequest,
|
| response_model=RecraftImageGenerationResponse,
|
| ),
|
| request=RecraftImageGenerationRequest(
|
| prompt=prompt,
|
| negative_prompt=negative_prompt,
|
| model=RecraftModel.recraftv3,
|
| size=size,
|
| n=n,
|
| style=recraft_style.style,
|
| substyle=recraft_style.substyle,
|
| style_id=recraft_style.style_id,
|
| controls=controls_api,
|
| ),
|
| auth_kwargs=kwargs,
|
| )
|
| response: RecraftImageGenerationResponse = operation.execute()
|
| images = []
|
| urls = []
|
| for data in response.data:
|
| with handle_recraft_image_output():
|
| if unique_id and data.url:
|
| urls.append(data.url)
|
| urls_string = '\n'.join(urls)
|
| PromptServer.instance.send_progress_text(
|
| f"Result URL: {urls_string}", unique_id
|
| )
|
| image = bytesio_to_image_tensor(
|
| download_url_to_bytesio(data.url, timeout=1024)
|
| )
|
| if len(image.shape) < 4:
|
| image = image.unsqueeze(0)
|
| images.append(image)
|
| output_image = torch.cat(images, dim=0)
|
|
|
| return (output_image,)
|
|
|
|
|
| class RecraftImageToImageNode:
|
| """
|
| Modify image based on prompt and strength.
|
| """
|
|
|
| RETURN_TYPES = (IO.IMAGE,)
|
| DESCRIPTION = cleandoc(__doc__ or "")
|
| FUNCTION = "api_call"
|
| API_NODE = True
|
| CATEGORY = "api node/image/Recraft"
|
|
|
| @classmethod
|
| def INPUT_TYPES(s):
|
| return {
|
| "required": {
|
| "image": (IO.IMAGE, ),
|
| "prompt": (
|
| IO.STRING,
|
| {
|
| "multiline": True,
|
| "default": "",
|
| "tooltip": "Prompt for the image generation.",
|
| },
|
| ),
|
| "n": (
|
| IO.INT,
|
| {
|
| "default": 1,
|
| "min": 1,
|
| "max": 6,
|
| "tooltip": "The number of images to generate.",
|
| },
|
| ),
|
| "strength": (
|
| IO.FLOAT,
|
| {
|
| "default": 0.5,
|
| "min": 0.0,
|
| "max": 1.0,
|
| "step": 0.01,
|
| "tooltip": "Defines the difference with the original image, should lie in [0, 1], where 0 means almost identical, and 1 means miserable similarity."
|
| }
|
| ),
|
| "seed": (
|
| IO.INT,
|
| {
|
| "default": 0,
|
| "min": 0,
|
| "max": 0xFFFFFFFFFFFFFFFF,
|
| "control_after_generate": True,
|
| "tooltip": "Seed to determine if node should re-run; actual results are nondeterministic regardless of seed.",
|
| },
|
| ),
|
| },
|
| "optional": {
|
| "recraft_style": (RecraftIO.STYLEV3,),
|
| "negative_prompt": (
|
| IO.STRING,
|
| {
|
| "default": "",
|
| "forceInput": True,
|
| "tooltip": "An optional text description of undesired elements on an image.",
|
| },
|
| ),
|
| "recraft_controls": (
|
| RecraftIO.CONTROLS,
|
| {
|
| "tooltip": "Optional additional controls over the generation via the Recraft Controls node."
|
| },
|
| ),
|
| },
|
| "hidden": {
|
| "auth_token": "AUTH_TOKEN_COMFY_ORG",
|
| "comfy_api_key": "API_KEY_COMFY_ORG",
|
| },
|
| }
|
|
|
| def api_call(
|
| self,
|
| image: torch.Tensor,
|
| prompt: str,
|
| n: int,
|
| strength: float,
|
| seed,
|
| recraft_style: RecraftStyle = None,
|
| negative_prompt: str = None,
|
| recraft_controls: RecraftControls = None,
|
| **kwargs,
|
| ):
|
| validate_string(prompt, strip_whitespace=False, max_length=1000)
|
| default_style = RecraftStyle(RecraftStyleV3.realistic_image)
|
| if recraft_style is None:
|
| recraft_style = default_style
|
|
|
| controls_api = None
|
| if recraft_controls:
|
| controls_api = recraft_controls.create_api_model()
|
|
|
| if not negative_prompt:
|
| negative_prompt = None
|
|
|
| request = RecraftImageGenerationRequest(
|
| prompt=prompt,
|
| negative_prompt=negative_prompt,
|
| model=RecraftModel.recraftv3,
|
| n=n,
|
| strength=round(strength, 2),
|
| style=recraft_style.style,
|
| substyle=recraft_style.substyle,
|
| style_id=recraft_style.style_id,
|
| controls=controls_api,
|
| )
|
|
|
| images = []
|
| total = image.shape[0]
|
| pbar = ProgressBar(total)
|
| for i in range(total):
|
| sub_bytes = handle_recraft_file_request(
|
| image=image[i],
|
| path="/proxy/recraft/images/imageToImage",
|
| request=request,
|
| auth_kwargs=kwargs,
|
| )
|
| with handle_recraft_image_output():
|
| images.append(torch.cat([bytesio_to_image_tensor(x) for x in sub_bytes], dim=0))
|
| pbar.update(1)
|
|
|
| images_tensor = torch.cat(images, dim=0)
|
| return (images_tensor, )
|
|
|
|
|
| class RecraftImageInpaintingNode:
|
| """
|
| Modify image based on prompt and mask.
|
| """
|
|
|
| RETURN_TYPES = (IO.IMAGE,)
|
| DESCRIPTION = cleandoc(__doc__ or "")
|
| FUNCTION = "api_call"
|
| API_NODE = True
|
| CATEGORY = "api node/image/Recraft"
|
|
|
| @classmethod
|
| def INPUT_TYPES(s):
|
| return {
|
| "required": {
|
| "image": (IO.IMAGE, ),
|
| "mask": (IO.MASK, ),
|
| "prompt": (
|
| IO.STRING,
|
| {
|
| "multiline": True,
|
| "default": "",
|
| "tooltip": "Prompt for the image generation.",
|
| },
|
| ),
|
| "n": (
|
| IO.INT,
|
| {
|
| "default": 1,
|
| "min": 1,
|
| "max": 6,
|
| "tooltip": "The number of images to generate.",
|
| },
|
| ),
|
| "seed": (
|
| IO.INT,
|
| {
|
| "default": 0,
|
| "min": 0,
|
| "max": 0xFFFFFFFFFFFFFFFF,
|
| "control_after_generate": True,
|
| "tooltip": "Seed to determine if node should re-run; actual results are nondeterministic regardless of seed.",
|
| },
|
| ),
|
| },
|
| "optional": {
|
| "recraft_style": (RecraftIO.STYLEV3,),
|
| "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",
|
| },
|
| }
|
|
|
| def api_call(
|
| self,
|
| image: torch.Tensor,
|
| mask: torch.Tensor,
|
| prompt: str,
|
| n: int,
|
| seed,
|
| recraft_style: RecraftStyle = None,
|
| negative_prompt: str = None,
|
| **kwargs,
|
| ):
|
| validate_string(prompt, strip_whitespace=False, max_length=1000)
|
| default_style = RecraftStyle(RecraftStyleV3.realistic_image)
|
| if recraft_style is None:
|
| recraft_style = default_style
|
|
|
| if not negative_prompt:
|
| negative_prompt = None
|
|
|
| request = RecraftImageGenerationRequest(
|
| prompt=prompt,
|
| negative_prompt=negative_prompt,
|
| model=RecraftModel.recraftv3,
|
| n=n,
|
| style=recraft_style.style,
|
| substyle=recraft_style.substyle,
|
| style_id=recraft_style.style_id,
|
| )
|
|
|
|
|
| mask = resize_mask_to_image(mask, image, allow_gradient=False, add_channel_dim=True)
|
|
|
| images = []
|
| total = image.shape[0]
|
| pbar = ProgressBar(total)
|
| for i in range(total):
|
| sub_bytes = handle_recraft_file_request(
|
| image=image[i],
|
| mask=mask[i:i+1],
|
| path="/proxy/recraft/images/inpaint",
|
| request=request,
|
| auth_kwargs=kwargs,
|
| )
|
| with handle_recraft_image_output():
|
| images.append(torch.cat([bytesio_to_image_tensor(x) for x in sub_bytes], dim=0))
|
| pbar.update(1)
|
|
|
| images_tensor = torch.cat(images, dim=0)
|
| return (images_tensor, )
|
|
|
|
|
| class RecraftTextToVectorNode:
|
| """
|
| Generates SVG synchronously based on prompt and resolution.
|
| """
|
|
|
| RETURN_TYPES = ("SVG",)
|
| DESCRIPTION = cleandoc(__doc__ or "") if 'cleandoc' in globals() else __doc__
|
| FUNCTION = "api_call"
|
| API_NODE = True
|
| CATEGORY = "api node/image/Recraft"
|
|
|
| @classmethod
|
| def INPUT_TYPES(s):
|
| return {
|
| "required": {
|
| "prompt": (
|
| IO.STRING,
|
| {
|
| "multiline": True,
|
| "default": "",
|
| "tooltip": "Prompt for the image generation.",
|
| },
|
| ),
|
| "substyle": (get_v3_substyles(RecraftStyleV3.vector_illustration),),
|
| "size": (
|
| [res.value for res in RecraftImageSize],
|
| {
|
| "default": RecraftImageSize.res_1024x1024,
|
| "tooltip": "The size of the generated image.",
|
| },
|
| ),
|
| "n": (
|
| IO.INT,
|
| {
|
| "default": 1,
|
| "min": 1,
|
| "max": 6,
|
| "tooltip": "The number of images to generate.",
|
| },
|
| ),
|
| "seed": (
|
| IO.INT,
|
| {
|
| "default": 0,
|
| "min": 0,
|
| "max": 0xFFFFFFFFFFFFFFFF,
|
| "control_after_generate": True,
|
| "tooltip": "Seed to determine if node should re-run; actual results are nondeterministic regardless of seed.",
|
| },
|
| ),
|
| },
|
| "optional": {
|
| "negative_prompt": (
|
| IO.STRING,
|
| {
|
| "default": "",
|
| "forceInput": True,
|
| "tooltip": "An optional text description of undesired elements on an image.",
|
| },
|
| ),
|
| "recraft_controls": (
|
| RecraftIO.CONTROLS,
|
| {
|
| "tooltip": "Optional additional controls over the generation via the Recraft Controls node."
|
| },
|
| ),
|
| },
|
| "hidden": {
|
| "auth_token": "AUTH_TOKEN_COMFY_ORG",
|
| "comfy_api_key": "API_KEY_COMFY_ORG",
|
| "unique_id": "UNIQUE_ID",
|
| },
|
| }
|
|
|
| def api_call(
|
| self,
|
| prompt: str,
|
| substyle: str,
|
| size: str,
|
| n: int,
|
| seed,
|
| negative_prompt: str = None,
|
| recraft_controls: RecraftControls = None,
|
| unique_id: Optional[str] = None,
|
| **kwargs,
|
| ):
|
| validate_string(prompt, strip_whitespace=False, max_length=1000)
|
|
|
| recraft_style = RecraftStyle(RecraftStyleV3.vector_illustration, substyle=substyle)
|
|
|
| controls_api = None
|
| if recraft_controls:
|
| controls_api = recraft_controls.create_api_model()
|
|
|
| if not negative_prompt:
|
| negative_prompt = None
|
|
|
| operation = SynchronousOperation(
|
| endpoint=ApiEndpoint(
|
| path="/proxy/recraft/image_generation",
|
| method=HttpMethod.POST,
|
| request_model=RecraftImageGenerationRequest,
|
| response_model=RecraftImageGenerationResponse,
|
| ),
|
| request=RecraftImageGenerationRequest(
|
| prompt=prompt,
|
| negative_prompt=negative_prompt,
|
| model=RecraftModel.recraftv3,
|
| size=size,
|
| n=n,
|
| style=recraft_style.style,
|
| substyle=recraft_style.substyle,
|
| controls=controls_api,
|
| ),
|
| auth_kwargs=kwargs,
|
| )
|
| response: RecraftImageGenerationResponse = operation.execute()
|
| svg_data = []
|
| urls = []
|
| for data in response.data:
|
| if unique_id and data.url:
|
| urls.append(data.url)
|
|
|
| PromptServer.instance.send_progress_text(
|
| f"Result URL: {' '.join(urls)}", unique_id
|
| )
|
| svg_data.append(download_url_to_bytesio(data.url, timeout=1024))
|
|
|
| return (SVG(svg_data),)
|
|
|
|
|
| class RecraftVectorizeImageNode:
|
| """
|
| Generates SVG synchronously from an input image.
|
| """
|
|
|
| RETURN_TYPES = ("SVG",)
|
| DESCRIPTION = cleandoc(__doc__ or "") if 'cleandoc' in globals() else __doc__
|
| FUNCTION = "api_call"
|
| API_NODE = True
|
| CATEGORY = "api node/image/Recraft"
|
|
|
| @classmethod
|
| def INPUT_TYPES(s):
|
| return {
|
| "required": {
|
| "image": (IO.IMAGE, ),
|
| },
|
| "optional": {
|
| },
|
| "hidden": {
|
| "auth_token": "AUTH_TOKEN_COMFY_ORG",
|
| "comfy_api_key": "API_KEY_COMFY_ORG",
|
| },
|
| }
|
|
|
| def api_call(
|
| self,
|
| image: torch.Tensor,
|
| **kwargs,
|
| ):
|
| svgs = []
|
| total = image.shape[0]
|
| pbar = ProgressBar(total)
|
| for i in range(total):
|
| sub_bytes = handle_recraft_file_request(
|
| image=image[i],
|
| path="/proxy/recraft/images/vectorize",
|
| auth_kwargs=kwargs,
|
| )
|
| svgs.append(SVG(sub_bytes))
|
| pbar.update(1)
|
|
|
| return (SVG.combine_all(svgs), )
|
|
|
|
|
| class RecraftReplaceBackgroundNode:
|
| """
|
| Replace background on image, based on provided prompt.
|
| """
|
|
|
| RETURN_TYPES = (IO.IMAGE,)
|
| DESCRIPTION = cleandoc(__doc__ or "")
|
| FUNCTION = "api_call"
|
| API_NODE = True
|
| CATEGORY = "api node/image/Recraft"
|
|
|
| @classmethod
|
| def INPUT_TYPES(s):
|
| return {
|
| "required": {
|
| "image": (IO.IMAGE, ),
|
| "prompt": (
|
| IO.STRING,
|
| {
|
| "multiline": True,
|
| "default": "",
|
| "tooltip": "Prompt for the image generation.",
|
| },
|
| ),
|
| "n": (
|
| IO.INT,
|
| {
|
| "default": 1,
|
| "min": 1,
|
| "max": 6,
|
| "tooltip": "The number of images to generate.",
|
| },
|
| ),
|
| "seed": (
|
| IO.INT,
|
| {
|
| "default": 0,
|
| "min": 0,
|
| "max": 0xFFFFFFFFFFFFFFFF,
|
| "control_after_generate": True,
|
| "tooltip": "Seed to determine if node should re-run; actual results are nondeterministic regardless of seed.",
|
| },
|
| ),
|
| },
|
| "optional": {
|
| "recraft_style": (RecraftIO.STYLEV3,),
|
| "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",
|
| },
|
| }
|
|
|
| def api_call(
|
| self,
|
| image: torch.Tensor,
|
| prompt: str,
|
| n: int,
|
| seed,
|
| recraft_style: RecraftStyle = None,
|
| negative_prompt: str = None,
|
| **kwargs,
|
| ):
|
| default_style = RecraftStyle(RecraftStyleV3.realistic_image)
|
| if recraft_style is None:
|
| recraft_style = default_style
|
|
|
| if not negative_prompt:
|
| negative_prompt = None
|
|
|
| request = RecraftImageGenerationRequest(
|
| prompt=prompt,
|
| negative_prompt=negative_prompt,
|
| model=RecraftModel.recraftv3,
|
| n=n,
|
| style=recraft_style.style,
|
| substyle=recraft_style.substyle,
|
| style_id=recraft_style.style_id,
|
| )
|
|
|
| images = []
|
| total = image.shape[0]
|
| pbar = ProgressBar(total)
|
| for i in range(total):
|
| sub_bytes = handle_recraft_file_request(
|
| image=image[i],
|
| path="/proxy/recraft/images/replaceBackground",
|
| request=request,
|
| auth_kwargs=kwargs,
|
| )
|
| images.append(torch.cat([bytesio_to_image_tensor(x) for x in sub_bytes], dim=0))
|
| pbar.update(1)
|
|
|
| images_tensor = torch.cat(images, dim=0)
|
| return (images_tensor, )
|
|
|
|
|
| class RecraftRemoveBackgroundNode:
|
| """
|
| Remove background from image, and return processed image and mask.
|
| """
|
|
|
| RETURN_TYPES = (IO.IMAGE, IO.MASK)
|
| DESCRIPTION = cleandoc(__doc__ or "")
|
| FUNCTION = "api_call"
|
| API_NODE = True
|
| CATEGORY = "api node/image/Recraft"
|
|
|
| @classmethod
|
| def INPUT_TYPES(s):
|
| return {
|
| "required": {
|
| "image": (IO.IMAGE, ),
|
| },
|
| "optional": {
|
| },
|
| "hidden": {
|
| "auth_token": "AUTH_TOKEN_COMFY_ORG",
|
| "comfy_api_key": "API_KEY_COMFY_ORG",
|
| },
|
| }
|
|
|
| def api_call(
|
| self,
|
| image: torch.Tensor,
|
| **kwargs,
|
| ):
|
| images = []
|
| total = image.shape[0]
|
| pbar = ProgressBar(total)
|
| for i in range(total):
|
| sub_bytes = handle_recraft_file_request(
|
| image=image[i],
|
| path="/proxy/recraft/images/removeBackground",
|
| auth_kwargs=kwargs,
|
| )
|
| images.append(torch.cat([bytesio_to_image_tensor(x) for x in sub_bytes], dim=0))
|
| pbar.update(1)
|
|
|
| images_tensor = torch.cat(images, dim=0)
|
|
|
| masks_tensor = images_tensor[:,:,:,-1:].squeeze(-1)
|
| return (images_tensor, masks_tensor)
|
|
|
|
|
| class RecraftCrispUpscaleNode:
|
| """
|
| Upscale image synchronously.
|
| Enhances a given raster image using ‘crisp upscale’ tool, increasing image resolution, making the image sharper and cleaner.
|
| """
|
|
|
| RETURN_TYPES = (IO.IMAGE,)
|
| DESCRIPTION = cleandoc(__doc__ or "")
|
| FUNCTION = "api_call"
|
| API_NODE = True
|
| CATEGORY = "api node/image/Recraft"
|
|
|
| RECRAFT_PATH = "/proxy/recraft/images/crispUpscale"
|
|
|
| @classmethod
|
| def INPUT_TYPES(s):
|
| return {
|
| "required": {
|
| "image": (IO.IMAGE, ),
|
| },
|
| "optional": {
|
| },
|
| "hidden": {
|
| "auth_token": "AUTH_TOKEN_COMFY_ORG",
|
| "comfy_api_key": "API_KEY_COMFY_ORG",
|
| },
|
| }
|
|
|
| def api_call(
|
| self,
|
| image: torch.Tensor,
|
| **kwargs,
|
| ):
|
| images = []
|
| total = image.shape[0]
|
| pbar = ProgressBar(total)
|
| for i in range(total):
|
| sub_bytes = handle_recraft_file_request(
|
| image=image[i],
|
| path=self.RECRAFT_PATH,
|
| auth_kwargs=kwargs,
|
| )
|
| images.append(torch.cat([bytesio_to_image_tensor(x) for x in sub_bytes], dim=0))
|
| pbar.update(1)
|
|
|
| images_tensor = torch.cat(images, dim=0)
|
| return (images_tensor,)
|
|
|
|
|
| class RecraftCreativeUpscaleNode(RecraftCrispUpscaleNode):
|
| """
|
| Upscale image synchronously.
|
| Enhances a given raster image using ‘creative upscale’ tool, boosting resolution with a focus on refining small details and faces.
|
| """
|
|
|
| RETURN_TYPES = (IO.IMAGE,)
|
| DESCRIPTION = cleandoc(__doc__ or "")
|
| FUNCTION = "api_call"
|
| API_NODE = True
|
| CATEGORY = "api node/image/Recraft"
|
|
|
| RECRAFT_PATH = "/proxy/recraft/images/creativeUpscale"
|
|
|
|
|
|
|
|
|
| NODE_CLASS_MAPPINGS = {
|
| "RecraftTextToImageNode": RecraftTextToImageNode,
|
| "RecraftImageToImageNode": RecraftImageToImageNode,
|
| "RecraftImageInpaintingNode": RecraftImageInpaintingNode,
|
| "RecraftTextToVectorNode": RecraftTextToVectorNode,
|
| "RecraftVectorizeImageNode": RecraftVectorizeImageNode,
|
| "RecraftRemoveBackgroundNode": RecraftRemoveBackgroundNode,
|
| "RecraftReplaceBackgroundNode": RecraftReplaceBackgroundNode,
|
| "RecraftCrispUpscaleNode": RecraftCrispUpscaleNode,
|
| "RecraftCreativeUpscaleNode": RecraftCreativeUpscaleNode,
|
| "RecraftStyleV3RealisticImage": RecraftStyleV3RealisticImageNode,
|
| "RecraftStyleV3DigitalIllustration": RecraftStyleV3DigitalIllustrationNode,
|
| "RecraftStyleV3LogoRaster": RecraftStyleV3LogoRasterNode,
|
| "RecraftStyleV3InfiniteStyleLibrary": RecraftStyleInfiniteStyleLibrary,
|
| "RecraftColorRGB": RecraftColorRGBNode,
|
| "RecraftControls": RecraftControlsNode,
|
| }
|
|
|
|
|
| NODE_DISPLAY_NAME_MAPPINGS = {
|
| "RecraftTextToImageNode": "Recraft Text to Image",
|
| "RecraftImageToImageNode": "Recraft Image to Image",
|
| "RecraftImageInpaintingNode": "Recraft Image Inpainting",
|
| "RecraftTextToVectorNode": "Recraft Text to Vector",
|
| "RecraftVectorizeImageNode": "Recraft Vectorize Image",
|
| "RecraftRemoveBackgroundNode": "Recraft Remove Background",
|
| "RecraftReplaceBackgroundNode": "Recraft Replace Background",
|
| "RecraftCrispUpscaleNode": "Recraft Crisp Upscale Image",
|
| "RecraftCreativeUpscaleNode": "Recraft Creative Upscale Image",
|
| "RecraftStyleV3RealisticImage": "Recraft Style - Realistic Image",
|
| "RecraftStyleV3DigitalIllustration": "Recraft Style - Digital Illustration",
|
| "RecraftStyleV3LogoRaster": "Recraft Style - Logo Raster",
|
| "RecraftStyleV3InfiniteStyleLibrary": "Recraft Style - Infinite Style Library",
|
| "RecraftColorRGB": "Recraft Color RGB",
|
| "RecraftControls": "Recraft Controls",
|
| }
|
|
|