| | import torch |
| | from .tools import VariantSupport |
| | from comfy_execution.graph_utils import GraphBuilder |
| |
|
| | class TestLazyMixImages: |
| | @classmethod |
| | def INPUT_TYPES(cls): |
| | return { |
| | "required": { |
| | "image1": ("IMAGE",{"lazy": True}), |
| | "image2": ("IMAGE",{"lazy": True}), |
| | "mask": ("MASK",), |
| | }, |
| | } |
| |
|
| | RETURN_TYPES = ("IMAGE",) |
| | FUNCTION = "mix" |
| |
|
| | CATEGORY = "Testing/Nodes" |
| |
|
| | def check_lazy_status(self, mask, image1, image2): |
| | mask_min = mask.min() |
| | mask_max = mask.max() |
| | needed = [] |
| | if image1 is None and (mask_min != 1.0 or mask_max != 1.0): |
| | needed.append("image1") |
| | if image2 is None and (mask_min != 0.0 or mask_max != 0.0): |
| | needed.append("image2") |
| | return needed |
| |
|
| | |
| | def mix(self, mask, image1, image2): |
| | mask_min = mask.min() |
| | mask_max = mask.max() |
| | if mask_min == 0.0 and mask_max == 0.0: |
| | return (image1,) |
| | elif mask_min == 1.0 and mask_max == 1.0: |
| | return (image2,) |
| |
|
| | if len(mask.shape) == 2: |
| | mask = mask.unsqueeze(0) |
| | if len(mask.shape) == 3: |
| | mask = mask.unsqueeze(3) |
| | if mask.shape[3] < image1.shape[3]: |
| | mask = mask.repeat(1, 1, 1, image1.shape[3]) |
| |
|
| | result = image1 * (1. - mask) + image2 * mask, |
| | return (result[0],) |
| |
|
| | class TestVariadicAverage: |
| | @classmethod |
| | def INPUT_TYPES(cls): |
| | return { |
| | "required": { |
| | "input1": ("IMAGE",), |
| | }, |
| | } |
| |
|
| | RETURN_TYPES = ("IMAGE",) |
| | FUNCTION = "variadic_average" |
| |
|
| | CATEGORY = "Testing/Nodes" |
| |
|
| | def variadic_average(self, input1, **kwargs): |
| | inputs = [input1] |
| | while 'input' + str(len(inputs) + 1) in kwargs: |
| | inputs.append(kwargs['input' + str(len(inputs) + 1)]) |
| | return (torch.stack(inputs).mean(dim=0),) |
| |
|
| |
|
| | class TestCustomIsChanged: |
| | @classmethod |
| | def INPUT_TYPES(cls): |
| | return { |
| | "required": { |
| | "image": ("IMAGE",), |
| | }, |
| | "optional": { |
| | "should_change": ("BOOL", {"default": False}), |
| | }, |
| | } |
| |
|
| | RETURN_TYPES = ("IMAGE",) |
| | FUNCTION = "custom_is_changed" |
| |
|
| | CATEGORY = "Testing/Nodes" |
| |
|
| | def custom_is_changed(self, image, should_change=False): |
| | return (image,) |
| | |
| | @classmethod |
| | def IS_CHANGED(cls, should_change=False, *args, **kwargs): |
| | if should_change: |
| | return float("NaN") |
| | else: |
| | return False |
| |
|
| | class TestIsChangedWithConstants: |
| | @classmethod |
| | def INPUT_TYPES(cls): |
| | return { |
| | "required": { |
| | "image": ("IMAGE",), |
| | "value": ("FLOAT", {"default": 1.0, "min": 0.0, "max": 10.0}), |
| | }, |
| | } |
| |
|
| | RETURN_TYPES = ("IMAGE",) |
| | FUNCTION = "custom_is_changed" |
| |
|
| | CATEGORY = "Testing/Nodes" |
| |
|
| | def custom_is_changed(self, image, value): |
| | return (image * value,) |
| | |
| | @classmethod |
| | def IS_CHANGED(cls, image, value): |
| | if image is None: |
| | return value |
| | else: |
| | return image.mean().item() * value |
| |
|
| | class TestCustomValidation1: |
| | @classmethod |
| | def INPUT_TYPES(cls): |
| | return { |
| | "required": { |
| | "input1": ("IMAGE,FLOAT",), |
| | "input2": ("IMAGE,FLOAT",), |
| | }, |
| | } |
| |
|
| | RETURN_TYPES = ("IMAGE",) |
| | FUNCTION = "custom_validation1" |
| |
|
| | CATEGORY = "Testing/Nodes" |
| |
|
| | def custom_validation1(self, input1, input2): |
| | if isinstance(input1, float) and isinstance(input2, float): |
| | result = torch.ones([1, 512, 512, 3]) * input1 * input2 |
| | else: |
| | result = input1 * input2 |
| | return (result,) |
| |
|
| | @classmethod |
| | def VALIDATE_INPUTS(cls, input1=None, input2=None): |
| | if input1 is not None: |
| | if not isinstance(input1, (torch.Tensor, float)): |
| | return f"Invalid type of input1: {type(input1)}" |
| | if input2 is not None: |
| | if not isinstance(input2, (torch.Tensor, float)): |
| | return f"Invalid type of input2: {type(input2)}" |
| |
|
| | return True |
| |
|
| | class TestCustomValidation2: |
| | @classmethod |
| | def INPUT_TYPES(cls): |
| | return { |
| | "required": { |
| | "input1": ("IMAGE,FLOAT",), |
| | "input2": ("IMAGE,FLOAT",), |
| | }, |
| | } |
| |
|
| | RETURN_TYPES = ("IMAGE",) |
| | FUNCTION = "custom_validation2" |
| |
|
| | CATEGORY = "Testing/Nodes" |
| |
|
| | def custom_validation2(self, input1, input2): |
| | if isinstance(input1, float) and isinstance(input2, float): |
| | result = torch.ones([1, 512, 512, 3]) * input1 * input2 |
| | else: |
| | result = input1 * input2 |
| | return (result,) |
| |
|
| | @classmethod |
| | def VALIDATE_INPUTS(cls, input_types, input1=None, input2=None): |
| | if input1 is not None: |
| | if not isinstance(input1, (torch.Tensor, float)): |
| | return f"Invalid type of input1: {type(input1)}" |
| | if input2 is not None: |
| | if not isinstance(input2, (torch.Tensor, float)): |
| | return f"Invalid type of input2: {type(input2)}" |
| |
|
| | if 'input1' in input_types: |
| | if input_types['input1'] not in ["IMAGE", "FLOAT"]: |
| | return f"Invalid type of input1: {input_types['input1']}" |
| | if 'input2' in input_types: |
| | if input_types['input2'] not in ["IMAGE", "FLOAT"]: |
| | return f"Invalid type of input2: {input_types['input2']}" |
| |
|
| | return True |
| |
|
| | @VariantSupport() |
| | class TestCustomValidation3: |
| | @classmethod |
| | def INPUT_TYPES(cls): |
| | return { |
| | "required": { |
| | "input1": ("IMAGE,FLOAT",), |
| | "input2": ("IMAGE,FLOAT",), |
| | }, |
| | } |
| |
|
| | RETURN_TYPES = ("IMAGE",) |
| | FUNCTION = "custom_validation3" |
| |
|
| | CATEGORY = "Testing/Nodes" |
| |
|
| | def custom_validation3(self, input1, input2): |
| | if isinstance(input1, float) and isinstance(input2, float): |
| | result = torch.ones([1, 512, 512, 3]) * input1 * input2 |
| | else: |
| | result = input1 * input2 |
| | return (result,) |
| |
|
| | class TestCustomValidation4: |
| | @classmethod |
| | def INPUT_TYPES(cls): |
| | return { |
| | "required": { |
| | "input1": ("FLOAT",), |
| | "input2": ("FLOAT",), |
| | }, |
| | } |
| |
|
| | RETURN_TYPES = ("IMAGE",) |
| | FUNCTION = "custom_validation4" |
| |
|
| | CATEGORY = "Testing/Nodes" |
| |
|
| | def custom_validation4(self, input1, input2): |
| | result = torch.ones([1, 512, 512, 3]) * input1 * input2 |
| | return (result,) |
| |
|
| | @classmethod |
| | def VALIDATE_INPUTS(cls, input1, input2): |
| | if input1 is not None: |
| | if not isinstance(input1, float): |
| | return f"Invalid type of input1: {type(input1)}" |
| | if input2 is not None: |
| | if not isinstance(input2, float): |
| | return f"Invalid type of input2: {type(input2)}" |
| |
|
| | return True |
| |
|
| | class TestCustomValidation5: |
| | @classmethod |
| | def INPUT_TYPES(cls): |
| | return { |
| | "required": { |
| | "input1": ("FLOAT", {"min": 0.0, "max": 1.0}), |
| | "input2": ("FLOAT", {"min": 0.0, "max": 1.0}), |
| | }, |
| | } |
| |
|
| | RETURN_TYPES = ("IMAGE",) |
| | FUNCTION = "custom_validation5" |
| |
|
| | CATEGORY = "Testing/Nodes" |
| |
|
| | def custom_validation5(self, input1, input2): |
| | value = input1 * input2 |
| | return (torch.ones([1, 512, 512, 3]) * value,) |
| |
|
| | @classmethod |
| | def VALIDATE_INPUTS(cls, **kwargs): |
| | if kwargs['input2'] == 7.0: |
| | return "7s are not allowed. I've never liked 7s." |
| | return True |
| |
|
| | class TestDynamicDependencyCycle: |
| | @classmethod |
| | def INPUT_TYPES(cls): |
| | return { |
| | "required": { |
| | "input1": ("IMAGE",), |
| | "input2": ("IMAGE",), |
| | }, |
| | } |
| |
|
| | RETURN_TYPES = ("IMAGE",) |
| | FUNCTION = "dynamic_dependency_cycle" |
| |
|
| | CATEGORY = "Testing/Nodes" |
| |
|
| | def dynamic_dependency_cycle(self, input1, input2): |
| | g = GraphBuilder() |
| | mask = g.node("StubMask", value=0.5, height=512, width=512, batch_size=1) |
| | mix1 = g.node("TestLazyMixImages", image1=input1, mask=mask.out(0)) |
| | mix2 = g.node("TestLazyMixImages", image1=mix1.out(0), image2=input2, mask=mask.out(0)) |
| |
|
| | |
| | mix1.set_input("image2", mix2.out(0)) |
| |
|
| | return { |
| | "result": (mix2.out(0),), |
| | "expand": g.finalize(), |
| | } |
| |
|
| | class TestMixedExpansionReturns: |
| | @classmethod |
| | def INPUT_TYPES(cls): |
| | return { |
| | "required": { |
| | "input1": ("FLOAT",), |
| | }, |
| | } |
| |
|
| | RETURN_TYPES = ("IMAGE","IMAGE") |
| | FUNCTION = "mixed_expansion_returns" |
| |
|
| | CATEGORY = "Testing/Nodes" |
| |
|
| | def mixed_expansion_returns(self, input1): |
| | white_image = torch.ones([1, 512, 512, 3]) |
| | if input1 <= 0.1: |
| | return (torch.ones([1, 512, 512, 3]) * 0.1, white_image) |
| | elif input1 <= 0.2: |
| | return { |
| | "result": (torch.ones([1, 512, 512, 3]) * 0.2, white_image), |
| | } |
| | else: |
| | g = GraphBuilder() |
| | mask = g.node("StubMask", value=0.3, height=512, width=512, batch_size=1) |
| | black = g.node("StubImage", content="BLACK", height=512, width=512, batch_size=1) |
| | white = g.node("StubImage", content="WHITE", height=512, width=512, batch_size=1) |
| | mix = g.node("TestLazyMixImages", image1=black.out(0), image2=white.out(0), mask=mask.out(0)) |
| | return { |
| | "result": (mix.out(0), white_image), |
| | "expand": g.finalize(), |
| | } |
| |
|
| | TEST_NODE_CLASS_MAPPINGS = { |
| | "TestLazyMixImages": TestLazyMixImages, |
| | "TestVariadicAverage": TestVariadicAverage, |
| | "TestCustomIsChanged": TestCustomIsChanged, |
| | "TestIsChangedWithConstants": TestIsChangedWithConstants, |
| | "TestCustomValidation1": TestCustomValidation1, |
| | "TestCustomValidation2": TestCustomValidation2, |
| | "TestCustomValidation3": TestCustomValidation3, |
| | "TestCustomValidation4": TestCustomValidation4, |
| | "TestCustomValidation5": TestCustomValidation5, |
| | "TestDynamicDependencyCycle": TestDynamicDependencyCycle, |
| | "TestMixedExpansionReturns": TestMixedExpansionReturns, |
| | } |
| |
|
| | TEST_NODE_DISPLAY_NAME_MAPPINGS = { |
| | "TestLazyMixImages": "Lazy Mix Images", |
| | "TestVariadicAverage": "Variadic Average", |
| | "TestCustomIsChanged": "Custom IsChanged", |
| | "TestIsChangedWithConstants": "IsChanged With Constants", |
| | "TestCustomValidation1": "Custom Validation 1", |
| | "TestCustomValidation2": "Custom Validation 2", |
| | "TestCustomValidation3": "Custom Validation 3", |
| | "TestCustomValidation4": "Custom Validation 4", |
| | "TestCustomValidation5": "Custom Validation 5", |
| | "TestDynamicDependencyCycle": "Dynamic Dependency Cycle", |
| | "TestMixedExpansionReturns": "Mixed Expansion Returns", |
| | } |
| |
|