| | import torch, struct, json
|
| | from io import BytesIO
|
| | import latent_preview, comfy
|
| | from server import PromptServer
|
| | from comfy.model_base import SDXL, SVD_img2vid, Flux, WAN21, Chroma
|
| | from comfy import samplers
|
| | import numpy as np
|
| | from math import ceil
|
| | from latent_preview import TAESDPreviewerImpl
|
| | from comfy_execution.utils import get_executing_context
|
| |
|
| | def slerp(val, low, high):
|
| | low_norm = low / torch.norm(low, dim=1, keepdim=True)
|
| | high_norm = high / torch.norm(high, dim=1, keepdim=True)
|
| | dot = (low_norm * high_norm).sum(1)
|
| | if dot.mean() > 0.9995:
|
| | return low * val + high * (1 - val)
|
| | omega = torch.acos(dot)
|
| | so = torch.sin(omega)
|
| | res = (torch.sin((1.0 - val) * omega) / so).unsqueeze(1) * low + (torch.sin(val * omega) / so).unsqueeze(1) * high
|
| | return res
|
| |
|
| | def swarm_partial_noise(seed, latent_image):
|
| | generator = torch.manual_seed(seed)
|
| | return torch.randn(latent_image.size(), dtype=latent_image.dtype, layout=latent_image.layout, generator=generator, device="cpu")
|
| |
|
| | def swarm_fixed_noise(seed, latent_image, var_seed, var_seed_strength):
|
| | noises = []
|
| | for i in range(latent_image.size()[0]):
|
| | if var_seed_strength > 0:
|
| | noise = swarm_partial_noise(seed, latent_image[i])
|
| | var_noise = swarm_partial_noise(var_seed + i, latent_image[i])
|
| | if noise.ndim == 4:
|
| | for j in range(noise.shape[1]):
|
| | noise[:, j] = slerp(var_seed_strength, noise[:, j], var_noise[:, j])
|
| | else:
|
| | noise = slerp(var_seed_strength, noise, var_noise)
|
| | else:
|
| | noise = swarm_partial_noise(seed + i, latent_image[i])
|
| | noises.append(noise)
|
| | return torch.stack(noises, dim=0)
|
| |
|
| | def get_preview_metadata():
|
| | executing_context = get_executing_context()
|
| | prompt_id = None
|
| | node_id = None
|
| | if executing_context is not None:
|
| | prompt_id = executing_context.prompt_id
|
| | node_id = executing_context.node_id
|
| | if prompt_id is None:
|
| | prompt_id = PromptServer.instance.last_prompt_id
|
| | if node_id is None:
|
| | node_id = PromptServer.instance.last_node_id
|
| | return {"node_id": node_id, "prompt_id": prompt_id, "display_node_id": node_id, "parent_node_id": node_id, "real_node_id": node_id}
|
| |
|
| | def swarm_send_extra_preview(id, image):
|
| | server = PromptServer.instance
|
| | metadata = get_preview_metadata()
|
| | metadata["mime_type"] = "image/jpeg"
|
| | metadata["id"] = id
|
| | metadata_json = json.dumps(metadata).encode('utf-8')
|
| | bytesIO = BytesIO()
|
| | image.save(bytesIO, format="JPEG", quality=90, compress_level=4)
|
| | image_bytes = bytesIO.getvalue()
|
| | combined_data = bytearray()
|
| | combined_data.extend(struct.pack(">I", len(metadata_json)))
|
| | combined_data.extend(metadata_json)
|
| | combined_data.extend(image_bytes)
|
| | server.send_sync(9999123, combined_data, sid=server.client_id)
|
| |
|
| | def swarm_send_animated_preview(id, images):
|
| | server = PromptServer.instance
|
| | bytesIO = BytesIO()
|
| | images[0].save(bytesIO, save_all=True, duration=int(1000.0/6), append_images=images[1 : len(images)], lossless=False, quality=60, method=0, format='WEBP')
|
| | bytesIO.seek(0)
|
| | image_bytes = bytesIO.getvalue()
|
| | metadata = get_preview_metadata()
|
| | metadata["mime_type"] = "image/webp"
|
| | metadata["id"] = id
|
| | metadata_json = json.dumps(metadata).encode('utf-8')
|
| | combined_data = bytearray()
|
| | combined_data.extend(struct.pack(">I", len(metadata_json)))
|
| | combined_data.extend(metadata_json)
|
| | combined_data.extend(image_bytes)
|
| | server.send_sync(9999123, combined_data, sid=server.client_id)
|
| |
|
| | def calculate_sigmas_scheduler(model, scheduler_name, steps, sigma_min, sigma_max, rho):
|
| | model_sampling = model.get_model_object("model_sampling")
|
| | if scheduler_name == "karras":
|
| | return comfy.k_diffusion.sampling.get_sigmas_karras(n=steps, sigma_min=sigma_min if sigma_min >= 0 else float(model_sampling.sigma_min), sigma_max=sigma_max if sigma_max >= 0 else float(model_sampling.sigma_max), rho=rho)
|
| | elif scheduler_name == "exponential":
|
| | return comfy.k_diffusion.sampling.get_sigmas_exponential(n=steps, sigma_min=sigma_min if sigma_min >= 0 else float(model_sampling.sigma_min), sigma_max=sigma_max if sigma_max >= 0 else float(model_sampling.sigma_max))
|
| | else:
|
| | return None
|
| |
|
| | def make_swarm_sampler_callback(steps, device, model, previews):
|
| | previewer = latent_preview.get_previewer(device, model.model.latent_format) if previews != "none" else None
|
| | pbar = comfy.utils.ProgressBar(steps)
|
| | def callback(step, x0, x, total_steps):
|
| | pbar.update_absolute(step + 1, total_steps, None)
|
| | if previewer:
|
| | if (step == 0 or (step < 3 and x0.ndim == 5 and x0.shape[1] > 8)) and not isinstance(previewer, TAESDPreviewerImpl):
|
| | x0 = x0.clone().cpu()
|
| | if x0.ndim == 5:
|
| |
|
| | x0 = x0[0].permute(1, 0, 2, 3)
|
| | x0 = torch.flip(x0, [0])
|
| | def do_preview(id, index):
|
| | preview_img = previewer.decode_latent_to_preview_image("JPEG", x0[index:index+1])
|
| | swarm_send_extra_preview(id, preview_img[1])
|
| | if previews == "iterate":
|
| | do_preview(0, step % x0.shape[0])
|
| | elif previews == "animate":
|
| | if x0.shape[0] == 1:
|
| | do_preview(0, 0)
|
| | else:
|
| | images = []
|
| | for i in range(x0.shape[0]):
|
| | preview_img = previewer.decode_latent_to_preview_image("JPEG", x0[i:i+1])
|
| | images.append(preview_img[1])
|
| | swarm_send_animated_preview(0, images)
|
| | elif previews == "default":
|
| | for i in range(x0.shape[0]):
|
| | preview_img = previewer.decode_latent_to_preview_image("JPEG", x0[i:i+1])
|
| | swarm_send_extra_preview(i, preview_img[1])
|
| | elif previews == "one":
|
| | do_preview(0, 0)
|
| | elif previews == "second":
|
| | do_preview(0, 1 % x0.shape[0])
|
| | return callback
|
| |
|
| |
|
| | def loglinear_interp(t_steps, num_steps):
|
| | """
|
| | Performs log-linear interpolation of a given array of decreasing numbers.
|
| | """
|
| | xs = np.linspace(0, 1, len(t_steps))
|
| | ys = np.log(t_steps[::-1])
|
| |
|
| | new_xs = np.linspace(0, 1, num_steps)
|
| | new_ys = np.interp(new_xs, xs, ys)
|
| |
|
| | interped_ys = np.exp(new_ys)[::-1].copy()
|
| | return interped_ys
|
| |
|
| | AYS_NOISE_LEVELS = {
|
| | "SD1": [14.6146412293, 6.4745760956, 3.8636745985, 2.6946151520, 1.8841921177, 1.3943805092, 0.9642583904, 0.6523686016, 0.3977456272, 0.1515232662, 0.0291671582],
|
| | "SDXL":[14.6146412293, 6.3184485287, 3.7681790315, 2.1811480769, 1.3405244945, 0.8620721141, 0.5550693289, 0.3798540708, 0.2332364134, 0.1114188177, 0.0291671582],
|
| | "SVD": [700.00, 54.5, 15.886, 7.977, 4.248, 1.789, 0.981, 0.403, 0.173, 0.034, 0.002],
|
| |
|
| | "Flux": [0.9968, 0.9886, 0.9819, 0.975, 0.966, 0.9471, 0.9158, 0.8287, 0.5512, 0.2808, 0.001],
|
| | "Wan": [1.0, 0.997, 0.995, 0.993, 0.991, 0.989, 0.987, 0.985, 0.98, 0.975, 0.973, 0.968, 0.96, 0.946, 0.927, 0.902, 0.864, 0.776, 0.539, 0.208, 0.001],
|
| |
|
| | "Chroma": [0.992, 0.99, 0.988, 0.985, 0.982, 0.978, 0.973, 0.968, 0.961, 0.953, 0.943, 0.931, 0.917, 0.9, 0.881, 0.858, 0.832, 0.802, 0.769, 0.731, 0.69, 0.646, 0.599, 0.55, 0.501, 0.451, 0.402, 0.355, 0.311, 0.27, 0.232, 0.199, 0.169, 0.143, 0.12, 0.101, 0.084, 0.07, 0.058, 0.048, 0.001]
|
| | }
|
| |
|
| | def split_latent_tensor(latent_tensor, tile_size=1024, scale_factor=8):
|
| | """Generate tiles for a given latent tensor, considering the scaling factor."""
|
| | latent_tile_size = tile_size // scale_factor
|
| | height, width = latent_tensor.shape[-2:]
|
| |
|
| |
|
| | num_tiles_x = ceil(width / latent_tile_size)
|
| | num_tiles_y = ceil(height / latent_tile_size)
|
| |
|
| |
|
| | if width % latent_tile_size == 0:
|
| | num_tiles_x += 1
|
| | if height % latent_tile_size == 0:
|
| | num_tiles_y += 1
|
| |
|
| |
|
| | overlap_x = 0 if num_tiles_x == 1 else (num_tiles_x * latent_tile_size - width) / (num_tiles_x - 1)
|
| | overlap_y = 0 if num_tiles_y == 1 else (num_tiles_y * latent_tile_size - height) / (num_tiles_y - 1)
|
| | if overlap_x < 32 and num_tiles_x > 1:
|
| | num_tiles_x += 1
|
| | overlap_x = (num_tiles_x * latent_tile_size - width) / (num_tiles_x - 1)
|
| | if overlap_y < 32 and num_tiles_y > 1:
|
| | num_tiles_y += 1
|
| | overlap_y = (num_tiles_y * latent_tile_size - height) / (num_tiles_y - 1)
|
| |
|
| | tiles = []
|
| |
|
| | for i in range(num_tiles_y):
|
| | for j in range(num_tiles_x):
|
| | x_start = j * latent_tile_size - j * overlap_x
|
| | y_start = i * latent_tile_size - i * overlap_y
|
| |
|
| |
|
| | x_start = round(x_start)
|
| | y_start = round(y_start)
|
| |
|
| |
|
| | tile_tensor = latent_tensor[..., y_start:y_start + latent_tile_size, x_start:x_start + latent_tile_size]
|
| | tiles.append(((x_start, y_start, x_start + latent_tile_size, y_start + latent_tile_size), tile_tensor))
|
| |
|
| | return tiles
|
| |
|
| | def stitch_latent_tensors(original_size, tiles, scale_factor=8):
|
| | """Stitch tiles together to create the final upscaled latent tensor with overlaps."""
|
| | result = torch.zeros(original_size)
|
| |
|
| |
|
| | sorted_tiles = sorted(tiles, key=lambda x: (x[0][1], x[0][0]))
|
| |
|
| |
|
| | current_row_upper = None
|
| |
|
| | for (left, upper, right, lower), tile in sorted_tiles:
|
| |
|
| |
|
| | if current_row_upper != upper:
|
| | current_row_upper = upper
|
| | first_tile_in_row = True
|
| | else:
|
| | first_tile_in_row = False
|
| |
|
| | tile_width = right - left
|
| | tile_height = lower - upper
|
| | feather = tile_width // 8
|
| |
|
| | mask = torch.ones_like(tile)
|
| |
|
| | if not first_tile_in_row:
|
| | for t in range(feather):
|
| | mask[..., :, t:t+1] *= (1.0 / feather) * (t + 1)
|
| |
|
| | if upper != 0:
|
| | for t in range(feather):
|
| | mask[..., t:t+1, :] *= (1.0 / feather) * (t + 1)
|
| |
|
| |
|
| | combined_area = tile * mask + result[..., upper:lower, left:right] * (1.0 - mask)
|
| | result[..., upper:lower, left:right] = combined_area
|
| |
|
| | return result
|
| |
|
| | class SwarmKSampler:
|
| | @classmethod
|
| | def INPUT_TYPES(s):
|
| | return {
|
| | "required": {
|
| | "model": ("MODEL",),
|
| | "noise_seed": ("INT", {"default": 0, "min": 0, "max": 0xffffffffffffffff}),
|
| | "steps": ("INT", {"default": 20, "min": 1, "max": 10000}),
|
| | "cfg": ("FLOAT", {"default": 8.0, "min": 0.0, "max": 100.0, "step": 0.5, "round": 0.001}),
|
| | "sampler_name": (comfy.samplers.KSampler.SAMPLERS, ),
|
| | "scheduler": (["turbo", "align_your_steps", "ltxv", "ltxv-image"] + comfy.samplers.KSampler.SCHEDULERS, ),
|
| | "positive": ("CONDITIONING", ),
|
| | "negative": ("CONDITIONING", ),
|
| | "latent_image": ("LATENT", ),
|
| | "start_at_step": ("INT", {"default": 0, "min": 0, "max": 10000}),
|
| | "end_at_step": ("INT", {"default": 10000, "min": 0, "max": 10000}),
|
| | "var_seed": ("INT", {"default": 0, "min": 0, "max": 0xffffffffffffffff}),
|
| | "var_seed_strength": ("FLOAT", {"default": 0.0, "min": 0.0, "max": 1.0, "step": 0.05, "round": 0.001}),
|
| | "sigma_max": ("FLOAT", {"default": -1, "min": -1.0, "max": 1000.0, "step":0.01, "round": False}),
|
| | "sigma_min": ("FLOAT", {"default": -1, "min": -1.0, "max": 1000.0, "step":0.01, "round": False}),
|
| | "rho": ("FLOAT", {"default": 7.0, "min": 0.0, "max": 100.0, "step":0.01, "round": False}),
|
| | "add_noise": (["enable", "disable"], ),
|
| | "return_with_leftover_noise": (["disable", "enable"], ),
|
| | "previews": (["default", "none", "one", "second", "iterate", "animate"], ),
|
| | "tile_sample": ("BOOLEAN", {"default": False}),
|
| | "tile_size": ("INT", {"default": 1024, "min": 256, "max": 4096}),
|
| | }
|
| | }
|
| |
|
| | CATEGORY = "SwarmUI/sampling"
|
| | RETURN_TYPES = ("LATENT",)
|
| | FUNCTION = "run_sampling"
|
| | DESCRIPTION = "Works like a vanilla Comfy KSamplerAdvanced, but with extra inputs for advanced features such as sigma scale, tiling, previews, etc."
|
| |
|
| | def sample(self, model, noise_seed, steps, cfg, sampler_name, scheduler, positive, negative, latent_image, start_at_step, end_at_step, var_seed, var_seed_strength, sigma_max, sigma_min, rho, add_noise, return_with_leftover_noise, previews):
|
| | device = comfy.model_management.get_torch_device()
|
| | latent_samples = latent_image["samples"]
|
| | latent_samples = comfy.sample.fix_empty_latent_channels(model, latent_samples)
|
| | disable_noise = add_noise == "disable"
|
| |
|
| | if disable_noise:
|
| | noise = torch.zeros(latent_samples.size(), dtype=latent_samples.dtype, layout=latent_samples.layout, device="cpu")
|
| | else:
|
| | noise = swarm_fixed_noise(noise_seed, latent_samples, var_seed, var_seed_strength)
|
| |
|
| | noise_mask = None
|
| | if "noise_mask" in latent_image:
|
| | noise_mask = latent_image["noise_mask"]
|
| |
|
| | sigmas = None
|
| | if scheduler == "turbo":
|
| | timesteps = torch.flip(torch.arange(1, 11) * 100 - 1, (0,))[:steps]
|
| | sigmas = model.model.model_sampling.sigma(timesteps)
|
| | sigmas = torch.cat([sigmas, sigmas.new_zeros([1])])
|
| | elif scheduler == "ltx" or scheduler == "ltxv-image":
|
| | from comfy_extras.nodes_lt import LTXVScheduler
|
| | sigmas = LTXVScheduler().get_sigmas(steps, 2.05, 0.95, True, 0.1, latent_image if scheduler == "ltxv-image" else None)[0]
|
| | elif scheduler == "align_your_steps":
|
| | if isinstance(model.model, SDXL):
|
| | model_type = "SDXL"
|
| | elif isinstance(model.model, SVD_img2vid):
|
| | model_type = "SVD"
|
| | elif isinstance(model.model, Flux):
|
| | model_type = "Flux"
|
| | elif isinstance(model.model, WAN21):
|
| | model_type = "Wan"
|
| | elif isinstance(model.model, Chroma):
|
| | model_type = "Chroma"
|
| | else:
|
| | print(f"AlignYourSteps: Unknown model type: {type(model.model)}, defaulting to SD1")
|
| | model_type = "SD1"
|
| | sigmas = AYS_NOISE_LEVELS[model_type][:]
|
| | if (steps + 1) != len(sigmas):
|
| | sigmas = loglinear_interp(sigmas, steps + 1)
|
| | sigmas[-1] = 0
|
| | sigmas = torch.FloatTensor(sigmas)
|
| | elif sigma_min >= 0 and sigma_max >= 0 and scheduler in ["karras", "exponential"]:
|
| | if sampler_name in ['dpm_2', 'dpm_2_ancestral']:
|
| | sigmas = calculate_sigmas_scheduler(model, scheduler, steps + 1, sigma_min, sigma_max, rho)
|
| | sigmas = torch.cat([sigmas[:-2], sigmas[-1:]])
|
| | else:
|
| | sigmas = calculate_sigmas_scheduler(model, scheduler, steps, sigma_min, sigma_max, rho)
|
| | sigmas = sigmas.to(device)
|
| |
|
| | out = latent_image.copy()
|
| | if steps > 0:
|
| | callback = make_swarm_sampler_callback(steps, device, model, previews)
|
| |
|
| | samples = comfy.sample.sample(model, noise, steps, cfg, sampler_name, scheduler, positive, negative, latent_samples,
|
| | denoise=1.0, disable_noise=disable_noise, start_step=start_at_step, last_step=end_at_step,
|
| | force_full_denoise=return_with_leftover_noise == "disable", noise_mask=noise_mask, sigmas=sigmas, callback=callback, seed=noise_seed)
|
| | out["samples"] = samples
|
| | return (out, )
|
| |
|
| |
|
| | def tiled_sample(self, model, noise_seed, steps, cfg, sampler_name, scheduler, positive, negative, latent_image, start_at_step, end_at_step, var_seed, var_seed_strength, sigma_max, sigma_min, rho, add_noise, return_with_leftover_noise, previews, tile_size):
|
| | out = latent_image.copy()
|
| |
|
| | latent_samples = latent_image["samples"]
|
| | tiles = split_latent_tensor(latent_samples, tile_size=tile_size)
|
| |
|
| | resampled_tiles = []
|
| | for coords, tile in tiles:
|
| | resampled_tile = self.sample(model, noise_seed, steps, cfg, sampler_name, scheduler, positive, negative, {"samples": tile}, start_at_step, end_at_step, var_seed, var_seed_strength, sigma_max, sigma_min, rho, add_noise, return_with_leftover_noise, previews)
|
| | resampled_tiles.append((coords, resampled_tile[0]["samples"]))
|
| |
|
| | result = stitch_latent_tensors(latent_samples.shape, resampled_tiles)
|
| | out["samples"] = result
|
| | return (out,)
|
| |
|
| | def run_sampling(self, model, noise_seed, steps, cfg, sampler_name, scheduler, positive, negative, latent_image, start_at_step, end_at_step, var_seed, var_seed_strength, sigma_max, sigma_min, rho, add_noise, return_with_leftover_noise, previews, tile_sample, tile_size):
|
| | if tile_sample:
|
| | return self.tiled_sample(model, noise_seed, steps, cfg, sampler_name, scheduler, positive, negative, latent_image, start_at_step, end_at_step, var_seed, var_seed_strength, sigma_max, sigma_min, rho, add_noise, return_with_leftover_noise, previews, tile_size)
|
| | else:
|
| | return self.sample(model, noise_seed, steps, cfg, sampler_name, scheduler, positive, negative, latent_image, start_at_step, end_at_step, var_seed, var_seed_strength, sigma_max, sigma_min, rho, add_noise, return_with_leftover_noise, previews)
|
| |
|
| | NODE_CLASS_MAPPINGS = {
|
| | "SwarmKSampler": SwarmKSampler,
|
| | }
|
| |
|