commit
409ca6f1dd
78
comfy-nodes/external_vid.py
Normal file
78
comfy-nodes/external_vid.py
Normal file
@ -0,0 +1,78 @@
|
|||||||
|
import os
|
||||||
|
import folder_paths
|
||||||
|
import uuid
|
||||||
|
|
||||||
|
from tqdm import tqdm
|
||||||
|
|
||||||
|
video_extensions = ["webm", "mp4", "mkv", "gif"]
|
||||||
|
|
||||||
|
|
||||||
|
class ComfyUIDeployExternalVideo:
|
||||||
|
@classmethod
|
||||||
|
def INPUT_TYPES(s):
|
||||||
|
input_dir = folder_paths.get_input_directory()
|
||||||
|
files = []
|
||||||
|
for f in os.listdir(input_dir):
|
||||||
|
if os.path.isfile(os.path.join(input_dir, f)):
|
||||||
|
file_parts = f.split(".")
|
||||||
|
if len(file_parts) > 1 and (file_parts[-1] in video_extensions):
|
||||||
|
files.append(f)
|
||||||
|
return {
|
||||||
|
"required": {
|
||||||
|
"input_id": (
|
||||||
|
"STRING",
|
||||||
|
{"multiline": False, "default": "input_video"},
|
||||||
|
),
|
||||||
|
},
|
||||||
|
"optional": {
|
||||||
|
"meta_batch": ("VHS_BatchManager",),
|
||||||
|
"default_value": (sorted(files),),
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
CATEGORY = "Video Helper Suite 🎥🅥🅗🅢"
|
||||||
|
|
||||||
|
RETURN_TYPES = ("STRING",)
|
||||||
|
RETURN_NAMES = ("video")
|
||||||
|
|
||||||
|
FUNCTION = "load_video"
|
||||||
|
|
||||||
|
def load_video(self, input_id, default_value):
|
||||||
|
input_dir = folder_paths.get_input_directory()
|
||||||
|
if input_id.startswith("http"):
|
||||||
|
import requests
|
||||||
|
|
||||||
|
print("Fetching video from URL: ", input_id)
|
||||||
|
response = requests.get(input_id, stream=True)
|
||||||
|
file_size = int(response.headers.get("Content-Length", 0))
|
||||||
|
file_extension = input_id.split(".")[-1].split("?")[
|
||||||
|
0
|
||||||
|
] # Extract extension and handle URLs with parameters
|
||||||
|
if file_extension not in video_extensions:
|
||||||
|
file_extension = ".mp4"
|
||||||
|
|
||||||
|
unique_filename = str(uuid.uuid4()) + "." + file_extension
|
||||||
|
video_path = os.path.join(input_dir, unique_filename)
|
||||||
|
chunk_size = 1024 # 1 Kibibyte
|
||||||
|
|
||||||
|
num_bars = int(file_size / chunk_size)
|
||||||
|
|
||||||
|
with open(video_path, "wb") as out_file:
|
||||||
|
for chunk in tqdm(
|
||||||
|
response.iter_content(chunk_size=chunk_size),
|
||||||
|
total=num_bars,
|
||||||
|
unit="KB",
|
||||||
|
desc="Downloading",
|
||||||
|
leave=True,
|
||||||
|
):
|
||||||
|
out_file.write(chunk)
|
||||||
|
else:
|
||||||
|
video_path = os.path.abspath(os.path.join(input_dir, default_value))
|
||||||
|
|
||||||
|
return (video_path,)
|
||||||
|
|
||||||
|
|
||||||
|
NODE_CLASS_MAPPINGS = {"ComfyUIDeployExternalVid": ComfyUIDeployExternalVideo}
|
||||||
|
NODE_DISPLAY_NAME_MAPPINGS = {
|
||||||
|
"ComfyUIDeployExternalVid": "External Video (ComfyUI Deploy) path"
|
||||||
|
}
|
@ -20,25 +20,32 @@ import server
|
|||||||
from tqdm import tqdm
|
from tqdm import tqdm
|
||||||
|
|
||||||
BIGMIN = -(2**53 - 1)
|
BIGMIN = -(2**53 - 1)
|
||||||
BIGMAX = (2**53-1)
|
BIGMAX = 2**53 - 1
|
||||||
|
|
||||||
DIMMAX = 8192
|
DIMMAX = 8192
|
||||||
|
|
||||||
|
|
||||||
def ffmpeg_suitability(path):
|
def ffmpeg_suitability(path):
|
||||||
try:
|
try:
|
||||||
version = subprocess.run([path, "-version"], check=True,
|
version = subprocess.run(
|
||||||
capture_output=True).stdout.decode("utf-8")
|
[path, "-version"], check=True, capture_output=True
|
||||||
|
).stdout.decode("utf-8")
|
||||||
except:
|
except:
|
||||||
return 0
|
return 0
|
||||||
score = 0
|
score = 0
|
||||||
# rough layout of the importance of various features
|
# rough layout of the importance of various features
|
||||||
simple_criterion = [("libvpx", 20),("264",10), ("265",3),
|
simple_criterion = [
|
||||||
("svtav1",5),("libopus", 1)]
|
("libvpx", 20),
|
||||||
|
("264", 10),
|
||||||
|
("265", 3),
|
||||||
|
("svtav1", 5),
|
||||||
|
("libopus", 1),
|
||||||
|
]
|
||||||
for criterion in simple_criterion:
|
for criterion in simple_criterion:
|
||||||
if version.find(criterion[0]) >= 0:
|
if version.find(criterion[0]) >= 0:
|
||||||
score += criterion[1]
|
score += criterion[1]
|
||||||
# obtain rough compile year from copyright information
|
# obtain rough compile year from copyright information
|
||||||
copyright_index = version.find('2000-2')
|
copyright_index = version.find("2000-2")
|
||||||
if copyright_index >= 0:
|
if copyright_index >= 0:
|
||||||
copyright_year = version[copyright_index + 6 : copyright_index + 9]
|
copyright_year = version[copyright_index + 6 : copyright_index + 9]
|
||||||
if copyright_year.isnumeric():
|
if copyright_year.isnumeric():
|
||||||
@ -52,6 +59,7 @@ else:
|
|||||||
ffmpeg_paths = []
|
ffmpeg_paths = []
|
||||||
try:
|
try:
|
||||||
from imageio_ffmpeg import get_ffmpeg_exe
|
from imageio_ffmpeg import get_ffmpeg_exe
|
||||||
|
|
||||||
imageio_ffmpeg_path = get_ffmpeg_exe()
|
imageio_ffmpeg_path = get_ffmpeg_exe()
|
||||||
ffmpeg_paths.append(imageio_ffmpeg_path)
|
ffmpeg_paths.append(imageio_ffmpeg_path)
|
||||||
except:
|
except:
|
||||||
@ -81,7 +89,13 @@ if gifski_path is None:
|
|||||||
if gifski_path is None:
|
if gifski_path is None:
|
||||||
gifski_path = shutil.which("gifski")
|
gifski_path = shutil.which("gifski")
|
||||||
|
|
||||||
def get_sorted_dir_files_from_directory(directory: str, skip_first_images: int=0, select_every_nth: int=1, extensions: Iterable=None):
|
|
||||||
|
def get_sorted_dir_files_from_directory(
|
||||||
|
directory: str,
|
||||||
|
skip_first_images: int = 0,
|
||||||
|
select_every_nth: int = 1,
|
||||||
|
extensions: Iterable = None,
|
||||||
|
):
|
||||||
directory = directory.strip()
|
directory = directory.strip()
|
||||||
dir_files = os.listdir(directory)
|
dir_files = os.listdir(directory)
|
||||||
dir_files = sorted(dir_files)
|
dir_files = sorted(dir_files)
|
||||||
@ -111,17 +125,24 @@ def calculate_file_hash(filename: str, hash_every_n: int = 1):
|
|||||||
h.update(str(os.path.getmtime(filename)).encode())
|
h.update(str(os.path.getmtime(filename)).encode())
|
||||||
return h.hexdigest()
|
return h.hexdigest()
|
||||||
|
|
||||||
|
|
||||||
prompt_queue = server.PromptServer.instance.prompt_queue
|
prompt_queue = server.PromptServer.instance.prompt_queue
|
||||||
|
|
||||||
|
|
||||||
def requeue_workflow_unchecked():
|
def requeue_workflow_unchecked():
|
||||||
"""Requeues the current workflow without checking for multiple requeues"""
|
"""Requeues the current workflow without checking for multiple requeues"""
|
||||||
currently_running = prompt_queue.currently_running
|
currently_running = prompt_queue.currently_running
|
||||||
(_, _, prompt, extra_data, outputs_to_execute) = next(iter(currently_running.values()))
|
(_, _, prompt, extra_data, outputs_to_execute) = next(
|
||||||
|
iter(currently_running.values())
|
||||||
|
)
|
||||||
|
|
||||||
# Ensure batch_managers are marked stale
|
# Ensure batch_managers are marked stale
|
||||||
prompt = prompt.copy()
|
prompt = prompt.copy()
|
||||||
for uid in prompt:
|
for uid in prompt:
|
||||||
if prompt[uid]['class_type'] == 'VHS_BatchManager':
|
if prompt[uid]["class_type"] == "VHS_BatchManager":
|
||||||
prompt[uid]['inputs']['requeue'] = prompt[uid]['inputs'].get('requeue',0)+1
|
prompt[uid]["inputs"]["requeue"] = (
|
||||||
|
prompt[uid]["inputs"].get("requeue", 0) + 1
|
||||||
|
)
|
||||||
|
|
||||||
# execution.py has guards for concurrency, but server doesn't.
|
# execution.py has guards for concurrency, but server doesn't.
|
||||||
# TODO: Check that this won't be an issue
|
# TODO: Check that this won't be an issue
|
||||||
@ -130,19 +151,22 @@ def requeue_workflow_unchecked():
|
|||||||
prompt_id = str(server.uuid.uuid4())
|
prompt_id = str(server.uuid.uuid4())
|
||||||
prompt_queue.put((number, prompt_id, prompt, extra_data, outputs_to_execute))
|
prompt_queue.put((number, prompt_id, prompt, extra_data, outputs_to_execute))
|
||||||
|
|
||||||
|
|
||||||
requeue_guard = [None, 0, 0, {}]
|
requeue_guard = [None, 0, 0, {}]
|
||||||
|
|
||||||
|
|
||||||
def requeue_workflow(requeue_required=(-1, True)):
|
def requeue_workflow(requeue_required=(-1, True)):
|
||||||
assert(len(prompt_queue.currently_running) == 1)
|
assert len(prompt_queue.currently_running) == 1
|
||||||
global requeue_guard
|
global requeue_guard
|
||||||
(run_number, _, prompt, _, _) = next(iter(prompt_queue.currently_running.values()))
|
(run_number, _, prompt, _, _) = next(iter(prompt_queue.currently_running.values()))
|
||||||
if requeue_guard[0] != run_number:
|
if requeue_guard[0] != run_number:
|
||||||
# Calculate a count of how many outputs are managed by a batch manager
|
# Calculate a count of how many outputs are managed by a batch manager
|
||||||
managed_outputs = 0
|
managed_outputs = 0
|
||||||
for bm_uid in prompt:
|
for bm_uid in prompt:
|
||||||
if prompt[bm_uid]['class_type'] == 'VHS_BatchManager':
|
if prompt[bm_uid]["class_type"] == "VHS_BatchManager":
|
||||||
for output_uid in prompt:
|
for output_uid in prompt:
|
||||||
if prompt[output_uid]['class_type'] in ["VHS_VideoCombine"]:
|
if prompt[output_uid]["class_type"] in ["VHS_VideoCombine"]:
|
||||||
for inp in prompt[output_uid]['inputs'].values():
|
for inp in prompt[output_uid]["inputs"].values():
|
||||||
if inp == [bm_uid, 0]:
|
if inp == [bm_uid, 0]:
|
||||||
managed_outputs += 1
|
managed_outputs += 1
|
||||||
requeue_guard = [run_number, 0, managed_outputs, {}]
|
requeue_guard = [run_number, 0, managed_outputs, {}]
|
||||||
@ -151,6 +175,7 @@ def requeue_workflow(requeue_required=(-1,True)):
|
|||||||
if requeue_guard[1] == requeue_guard[2] and max(requeue_guard[3].values()):
|
if requeue_guard[1] == requeue_guard[2] and max(requeue_guard[3].values()):
|
||||||
requeue_workflow_unchecked()
|
requeue_workflow_unchecked()
|
||||||
|
|
||||||
|
|
||||||
def get_audio(file, start_time=0, duration=0):
|
def get_audio(file, start_time=0, duration=0):
|
||||||
args = [ffmpeg_path, "-v", "error", "-i", file]
|
args = [ffmpeg_path, "-v", "error", "-i", file]
|
||||||
if start_time > 0:
|
if start_time > 0:
|
||||||
@ -158,8 +183,9 @@ def get_audio(file, start_time=0, duration=0):
|
|||||||
if duration > 0:
|
if duration > 0:
|
||||||
args += ["-t", str(duration)]
|
args += ["-t", str(duration)]
|
||||||
try:
|
try:
|
||||||
res = subprocess.run(args + ["-f", "wav", "-"],
|
res = subprocess.run(
|
||||||
stdout=subprocess.PIPE, check=True).stdout
|
args + ["-f", "wav", "-"], stdout=subprocess.PIPE, check=True
|
||||||
|
).stdout
|
||||||
except subprocess.CalledProcessError as e:
|
except subprocess.CalledProcessError as e:
|
||||||
return False
|
return False
|
||||||
return res
|
return res
|
||||||
@ -170,10 +196,12 @@ def lazy_eval(func):
|
|||||||
def __init__(self, func):
|
def __init__(self, func):
|
||||||
self.res = None
|
self.res = None
|
||||||
self.func = func
|
self.func = func
|
||||||
|
|
||||||
def get(self):
|
def get(self):
|
||||||
if self.res is None:
|
if self.res is None:
|
||||||
self.res = self.func()
|
self.res = self.func()
|
||||||
return self.res
|
return self.res
|
||||||
|
|
||||||
cache = Cache(func)
|
cache = Cache(func)
|
||||||
return lambda: cache.get()
|
return lambda: cache.get()
|
||||||
|
|
||||||
@ -181,31 +209,33 @@ def lazy_eval(func):
|
|||||||
def is_url(url):
|
def is_url(url):
|
||||||
return url.split("://")[0] in ["http", "https"]
|
return url.split("://")[0] in ["http", "https"]
|
||||||
|
|
||||||
|
|
||||||
def validate_sequence(path):
|
def validate_sequence(path):
|
||||||
# Check if path is a valid ffmpeg sequence that points to at least one file
|
# Check if path is a valid ffmpeg sequence that points to at least one file
|
||||||
(path, file) = os.path.split(path)
|
(path, file) = os.path.split(path)
|
||||||
if not os.path.isdir(path):
|
if not os.path.isdir(path):
|
||||||
return False
|
return False
|
||||||
match = re.search('%0?\d+d', file)
|
match = re.search("%0?\d+d", file)
|
||||||
if not match:
|
if not match:
|
||||||
return False
|
return False
|
||||||
seq = match.group()
|
seq = match.group()
|
||||||
if seq == '%d':
|
if seq == "%d":
|
||||||
seq = '\\\\d+'
|
seq = "\\\\d+"
|
||||||
else:
|
else:
|
||||||
seq = '\\\\d{%s}' % seq[1:-1]
|
seq = "\\\\d{%s}" % seq[1:-1]
|
||||||
file_matcher = re.compile(re.sub('%0?\d+d', seq, file))
|
file_matcher = re.compile(re.sub("%0?\d+d", seq, file))
|
||||||
for file in os.listdir(path):
|
for file in os.listdir(path):
|
||||||
if file_matcher.fullmatch(file):
|
if file_matcher.fullmatch(file):
|
||||||
return True
|
return True
|
||||||
return False
|
return False
|
||||||
|
|
||||||
|
|
||||||
def hash_path(path):
|
def hash_path(path):
|
||||||
if path is None:
|
if path is None:
|
||||||
return "input"
|
return "input"
|
||||||
if is_url(path):
|
if is_url(path):
|
||||||
return "url"
|
return "url"
|
||||||
return calculate_file_hash(path.strip("\""))
|
return calculate_file_hash(path.strip('"'))
|
||||||
|
|
||||||
|
|
||||||
def validate_path(path, allow_none=False, allow_url=True):
|
def validate_path(path, allow_none=False, allow_url=True):
|
||||||
@ -214,21 +244,24 @@ def validate_path(path, allow_none=False, allow_url=True):
|
|||||||
if is_url(path):
|
if is_url(path):
|
||||||
# Probably not feasible to check if url resolves here
|
# Probably not feasible to check if url resolves here
|
||||||
return True if allow_url else "URLs are unsupported for this path"
|
return True if allow_url else "URLs are unsupported for this path"
|
||||||
if not os.path.isfile(path.strip("\"")):
|
if not os.path.isfile(path.strip('"')):
|
||||||
return "Invalid file path: {}".format(path)
|
return "Invalid file path: {}".format(path)
|
||||||
return True
|
return True
|
||||||
|
|
||||||
|
|
||||||
### Utils
|
### Utils
|
||||||
|
|
||||||
video_extensions = ['webm', 'mp4', 'mkv', 'gif']
|
video_extensions = ["webm", "mp4", "mkv", "gif"]
|
||||||
|
|
||||||
|
|
||||||
def is_gif(filename) -> bool:
|
def is_gif(filename) -> bool:
|
||||||
file_parts = filename.split('.')
|
file_parts = filename.split(".")
|
||||||
return len(file_parts) > 1 and file_parts[-1] == "gif"
|
return len(file_parts) > 1 and file_parts[-1] == "gif"
|
||||||
|
|
||||||
|
|
||||||
def target_size(width, height, force_size, custom_width, custom_height) -> tuple[int, int]:
|
def target_size(
|
||||||
|
width, height, force_size, custom_width, custom_height
|
||||||
|
) -> tuple[int, int]:
|
||||||
if force_size == "Custom":
|
if force_size == "Custom":
|
||||||
return (custom_width, custom_height)
|
return (custom_width, custom_height)
|
||||||
elif force_size == "Custom Height":
|
elif force_size == "Custom Height":
|
||||||
@ -252,8 +285,16 @@ def target_size(width, height, force_size, custom_width, custom_height) -> tuple
|
|||||||
height = int(force_size[1])
|
height = int(force_size[1])
|
||||||
return (width, height)
|
return (width, height)
|
||||||
|
|
||||||
def cv_frame_generator(video, force_rate, frame_load_cap, skip_first_frames,
|
|
||||||
select_every_nth, meta_batch=None, unique_id=None):
|
def cv_frame_generator(
|
||||||
|
video,
|
||||||
|
force_rate,
|
||||||
|
frame_load_cap,
|
||||||
|
skip_first_frames,
|
||||||
|
select_every_nth,
|
||||||
|
meta_batch=None,
|
||||||
|
unique_id=None,
|
||||||
|
):
|
||||||
video_cap = cv2.VideoCapture(video)
|
video_cap = cv2.VideoCapture(video)
|
||||||
if not video_cap.isOpened():
|
if not video_cap.isOpened():
|
||||||
raise ValueError(f"{video} could not be loaded with cv.")
|
raise ValueError(f"{video} could not be loaded with cv.")
|
||||||
@ -325,26 +366,54 @@ def cv_frame_generator(video, force_rate, frame_load_cap, skip_first_frames,
|
|||||||
if prev_frame is not None:
|
if prev_frame is not None:
|
||||||
yield prev_frame
|
yield prev_frame
|
||||||
|
|
||||||
def load_video_cv(video: str, force_rate: int, force_size: str,
|
|
||||||
custom_width: int,custom_height: int, frame_load_cap: int,
|
def load_video_cv(
|
||||||
skip_first_frames: int, select_every_nth: int,
|
video: str,
|
||||||
meta_batch=None, unique_id=None):
|
force_rate: int,
|
||||||
|
force_size: str,
|
||||||
|
custom_width: int,
|
||||||
|
custom_height: int,
|
||||||
|
frame_load_cap: int,
|
||||||
|
skip_first_frames: int,
|
||||||
|
select_every_nth: int,
|
||||||
|
meta_batch=None,
|
||||||
|
unique_id=None,
|
||||||
|
):
|
||||||
if meta_batch is None or unique_id not in meta_batch.inputs:
|
if meta_batch is None or unique_id not in meta_batch.inputs:
|
||||||
gen = cv_frame_generator(video, force_rate, frame_load_cap, skip_first_frames,
|
gen = cv_frame_generator(
|
||||||
select_every_nth, meta_batch, unique_id)
|
video,
|
||||||
|
force_rate,
|
||||||
|
frame_load_cap,
|
||||||
|
skip_first_frames,
|
||||||
|
select_every_nth,
|
||||||
|
meta_batch,
|
||||||
|
unique_id,
|
||||||
|
)
|
||||||
(width, height, fps, duration, total_frames, target_frame_time) = next(gen)
|
(width, height, fps, duration, total_frames, target_frame_time) = next(gen)
|
||||||
|
|
||||||
if meta_batch is not None:
|
if meta_batch is not None:
|
||||||
meta_batch.inputs[unique_id] = (gen, width, height, fps, duration, total_frames, target_frame_time)
|
meta_batch.inputs[unique_id] = (
|
||||||
|
gen,
|
||||||
|
width,
|
||||||
|
height,
|
||||||
|
fps,
|
||||||
|
duration,
|
||||||
|
total_frames,
|
||||||
|
target_frame_time,
|
||||||
|
)
|
||||||
|
|
||||||
else:
|
else:
|
||||||
(gen, width, height, fps, duration, total_frames, target_frame_time) = meta_batch.inputs[unique_id]
|
(gen, width, height, fps, duration, total_frames, target_frame_time) = (
|
||||||
|
meta_batch.inputs[unique_id]
|
||||||
|
)
|
||||||
|
|
||||||
if meta_batch is not None:
|
if meta_batch is not None:
|
||||||
gen = itertools.islice(gen, meta_batch.frames_per_batch)
|
gen = itertools.islice(gen, meta_batch.frames_per_batch)
|
||||||
|
|
||||||
# Some minor wizardry to eliminate a copy and reduce max memory by a factor of ~2
|
# Some minor wizardry to eliminate a copy and reduce max memory by a factor of ~2
|
||||||
images = torch.from_numpy(np.fromiter(gen, np.dtype((np.float32, (height, width, 3)))))
|
images = torch.from_numpy(
|
||||||
|
np.fromiter(gen, np.dtype((np.float32, (height, width, 3))))
|
||||||
|
)
|
||||||
if len(images) == 0:
|
if len(images) == 0:
|
||||||
raise RuntimeError("No frames generated")
|
raise RuntimeError("No frames generated")
|
||||||
if force_size != "Disabled":
|
if force_size != "Disabled":
|
||||||
@ -355,8 +424,11 @@ def load_video_cv(video: str, force_rate: int, force_size: str,
|
|||||||
images = s.movedim(1, -1)
|
images = s.movedim(1, -1)
|
||||||
|
|
||||||
# Setup lambda for lazy audio capture
|
# Setup lambda for lazy audio capture
|
||||||
audio = lambda : get_audio(video, skip_first_frames * target_frame_time,
|
audio = lambda: get_audio(
|
||||||
frame_load_cap*target_frame_time*select_every_nth)
|
video,
|
||||||
|
skip_first_frames * target_frame_time,
|
||||||
|
frame_load_cap * target_frame_time * select_every_nth,
|
||||||
|
)
|
||||||
# Adjust target_frame_time for select_every_nth
|
# Adjust target_frame_time for select_every_nth
|
||||||
target_frame_time *= select_every_nth
|
target_frame_time *= select_every_nth
|
||||||
video_info = {
|
video_info = {
|
||||||
@ -382,60 +454,100 @@ class ComfyUIDeployExternalVideo:
|
|||||||
files = []
|
files = []
|
||||||
for f in os.listdir(input_dir):
|
for f in os.listdir(input_dir):
|
||||||
if os.path.isfile(os.path.join(input_dir, f)):
|
if os.path.isfile(os.path.join(input_dir, f)):
|
||||||
file_parts = f.split('.')
|
file_parts = f.split(".")
|
||||||
if len(file_parts) > 1 and (file_parts[-1] in video_extensions):
|
if len(file_parts) > 1 and (file_parts[-1] in video_extensions):
|
||||||
files.append(f)
|
files.append(f)
|
||||||
return {"required": {
|
return {
|
||||||
|
"required": {
|
||||||
"input_id": (
|
"input_id": (
|
||||||
"STRING",
|
"STRING",
|
||||||
{"multiline": False, "default": "input_video"},
|
{"multiline": False, "default": "input_video"},
|
||||||
),
|
),
|
||||||
"force_rate": ("INT", {"default": 0, "min": 0, "max": 60, "step": 1}),
|
"force_rate": ("INT", {"default": 0, "min": 0, "max": 60, "step": 1}),
|
||||||
"force_size": (["Disabled", "Custom Height", "Custom Width", "Custom", "256x?", "?x256", "256x256", "512x?", "?x512", "512x512"],),
|
"force_size": (
|
||||||
"custom_width": ("INT", {"default": 512, "min": 0, "max": DIMMAX, "step": 8}),
|
[
|
||||||
"custom_height": ("INT", {"default": 512, "min": 0, "max": DIMMAX, "step": 8}),
|
"Disabled",
|
||||||
"frame_load_cap": ("INT", {"default": 0, "min": 0, "max": BIGMAX, "step": 1}),
|
"Custom Height",
|
||||||
"skip_first_frames": ("INT", {"default": 0, "min": 0, "max": BIGMAX, "step": 1}),
|
"Custom Width",
|
||||||
"select_every_nth": ("INT", {"default": 1, "min": 1, "max": BIGMAX, "step": 1}),
|
"Custom",
|
||||||
|
"256x?",
|
||||||
|
"?x256",
|
||||||
|
"256x256",
|
||||||
|
"512x?",
|
||||||
|
"?x512",
|
||||||
|
"512x512",
|
||||||
|
],
|
||||||
|
),
|
||||||
|
"custom_width": (
|
||||||
|
"INT",
|
||||||
|
{"default": 512, "min": 0, "max": DIMMAX, "step": 8},
|
||||||
|
),
|
||||||
|
"custom_height": (
|
||||||
|
"INT",
|
||||||
|
{"default": 512, "min": 0, "max": DIMMAX, "step": 8},
|
||||||
|
),
|
||||||
|
"frame_load_cap": (
|
||||||
|
"INT",
|
||||||
|
{"default": 0, "min": 0, "max": BIGMAX, "step": 1},
|
||||||
|
),
|
||||||
|
"skip_first_frames": (
|
||||||
|
"INT",
|
||||||
|
{"default": 0, "min": 0, "max": BIGMAX, "step": 1},
|
||||||
|
),
|
||||||
|
"select_every_nth": (
|
||||||
|
"INT",
|
||||||
|
{"default": 1, "min": 1, "max": BIGMAX, "step": 1},
|
||||||
|
),
|
||||||
},
|
},
|
||||||
"optional": {
|
"optional": {
|
||||||
"meta_batch": ("VHS_BatchManager",),
|
"meta_batch": ("VHS_BatchManager",),
|
||||||
"default_value": (sorted(files),),
|
"default_value": (sorted(files),),
|
||||||
},
|
},
|
||||||
"hidden": {
|
"hidden": {"unique_id": "UNIQUE_ID"},
|
||||||
"unique_id": "UNIQUE_ID"
|
|
||||||
},
|
|
||||||
}
|
}
|
||||||
|
|
||||||
CATEGORY = "Video Helper Suite 🎥🅥🅗🅢"
|
CATEGORY = "Video Helper Suite 🎥🅥🅗🅢"
|
||||||
|
|
||||||
RETURN_TYPES = ("IMAGE", "INT", "VHS_AUDIO", "VHS_VIDEOINFO",)
|
RETURN_TYPES = (
|
||||||
RETURN_NAMES = ("IMAGE", "frame_count", "audio", "video_info",)
|
"IMAGE",
|
||||||
|
"INT",
|
||||||
|
"VHS_AUDIO",
|
||||||
|
"VHS_VIDEOINFO",
|
||||||
|
)
|
||||||
|
RETURN_NAMES = (
|
||||||
|
"IMAGE",
|
||||||
|
"frame_count",
|
||||||
|
"audio",
|
||||||
|
"video_info",
|
||||||
|
)
|
||||||
|
|
||||||
FUNCTION = "load_video"
|
FUNCTION = "load_video"
|
||||||
|
|
||||||
def load_video(self, **kwargs):
|
def load_video(self, **kwargs):
|
||||||
input_id = kwargs.get('input_id')
|
input_id = kwargs.get("input_id")
|
||||||
force_rate = kwargs.get('force_rate')
|
force_rate = kwargs.get("force_rate")
|
||||||
force_size = kwargs.get('force_size', "Disabled")
|
force_size = kwargs.get("force_size", "Disabled")
|
||||||
custom_width = kwargs.get('custom_width')
|
custom_width = kwargs.get("custom_width")
|
||||||
custom_height = kwargs.get('custom_height')
|
custom_height = kwargs.get("custom_height")
|
||||||
frame_load_cap = kwargs.get('frame_load_cap')
|
frame_load_cap = kwargs.get("frame_load_cap")
|
||||||
skip_first_frames = kwargs.get('skip_first_frames')
|
skip_first_frames = kwargs.get("skip_first_frames")
|
||||||
select_every_nth = kwargs.get('select_every_nth')
|
select_every_nth = kwargs.get("select_every_nth")
|
||||||
meta_batch = kwargs.get('meta_batch')
|
meta_batch = kwargs.get("meta_batch")
|
||||||
unique_id = kwargs.get('unique_id')
|
unique_id = kwargs.get("unique_id")
|
||||||
|
|
||||||
video = kwargs.get('default_value')
|
video = kwargs.get("default_value")
|
||||||
video_path = folder_paths.get_annotated_filepath(video.strip("\""))
|
video_path = folder_paths.get_annotated_filepath(video.strip('"'))
|
||||||
|
|
||||||
input_dir = folder_paths.get_input_directory()
|
input_dir = folder_paths.get_input_directory()
|
||||||
if input_id.startswith('http'):
|
if input_id.startswith("http"):
|
||||||
import requests
|
import requests
|
||||||
|
|
||||||
print("Fetching video from URL: ", input_id)
|
print("Fetching video from URL: ", input_id)
|
||||||
response = requests.get(input_id, stream=True)
|
response = requests.get(input_id, stream=True)
|
||||||
file_size = int(response.headers.get('Content-Length', 0))
|
file_size = int(response.headers.get("Content-Length", 0))
|
||||||
file_extension = input_id.split('.')[-1].split('?')[0] # Extract extension and handle URLs with parameters
|
file_extension = input_id.split(".")[-1].split("?")[
|
||||||
|
0
|
||||||
|
] # Extract extension and handle URLs with parameters
|
||||||
if file_extension not in video_extensions:
|
if file_extension not in video_extensions:
|
||||||
file_extension = ".mp4"
|
file_extension = ".mp4"
|
||||||
|
|
||||||
@ -445,27 +557,38 @@ class ComfyUIDeployExternalVideo:
|
|||||||
|
|
||||||
num_bars = int(file_size / chunk_size)
|
num_bars = int(file_size / chunk_size)
|
||||||
|
|
||||||
with open(video_path, 'wb') as out_file:
|
with open(video_path, "wb") as out_file:
|
||||||
for chunk in tqdm(
|
for chunk in tqdm(
|
||||||
response.iter_content(chunk_size=chunk_size),
|
response.iter_content(chunk_size=chunk_size),
|
||||||
total=num_bars,
|
total=num_bars,
|
||||||
unit='KB',
|
unit="KB",
|
||||||
desc="Downloading",
|
desc="Downloading",
|
||||||
leave=True
|
leave=True,
|
||||||
):
|
):
|
||||||
out_file.write(chunk)
|
out_file.write(chunk)
|
||||||
|
|
||||||
print("video path: ", video_path)
|
print("video path: ", video_path)
|
||||||
|
|
||||||
return load_video_cv(video=video_path, force_rate=force_rate, force_size=force_size,
|
return load_video_cv(
|
||||||
custom_width=custom_width, custom_height=custom_height, frame_load_cap=frame_load_cap,
|
video=video_path,
|
||||||
skip_first_frames=skip_first_frames, select_every_nth=select_every_nth,
|
force_rate=force_rate,
|
||||||
meta_batch=meta_batch, unique_id=unique_id)
|
force_size=force_size,
|
||||||
|
custom_width=custom_width,
|
||||||
|
custom_height=custom_height,
|
||||||
|
frame_load_cap=frame_load_cap,
|
||||||
|
skip_first_frames=skip_first_frames,
|
||||||
|
select_every_nth=select_every_nth,
|
||||||
|
meta_batch=meta_batch,
|
||||||
|
unique_id=unique_id,
|
||||||
|
)
|
||||||
|
|
||||||
@classmethod
|
@classmethod
|
||||||
def IS_CHANGED(s, video, **kwargs):
|
def IS_CHANGED(s, video, **kwargs):
|
||||||
image_path = folder_paths.get_annotated_filepath(video)
|
image_path = folder_paths.get_annotated_filepath(video)
|
||||||
return calculate_file_hash(image_path)
|
return calculate_file_hash(image_path)
|
||||||
|
|
||||||
|
|
||||||
NODE_CLASS_MAPPINGS = {"ComfyUIDeployExternalVideo": ComfyUIDeployExternalVideo}
|
NODE_CLASS_MAPPINGS = {"ComfyUIDeployExternalVideo": ComfyUIDeployExternalVideo}
|
||||||
NODE_DISPLAY_NAME_MAPPINGS = {"ComfyUIDeployExternalVideo": "External Video (ComfyUI Deploy)"}
|
NODE_DISPLAY_NAME_MAPPINGS = {
|
||||||
|
"ComfyUIDeployExternalVideo": "External Video (ComfyUI Deploy x VHS)"
|
||||||
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user