|
|
import server |
|
|
import folder_paths |
|
|
import os |
|
|
import subprocess |
|
|
import re |
|
|
|
|
|
import asyncio |
|
|
import av |
|
|
|
|
|
from .utils import is_url, get_sorted_dir_files_from_directory, ffmpeg_path, \ |
|
|
validate_sequence, is_safe_path, strip_path, try_download_video, ENCODE_ARGS |
|
|
from comfy.k_diffusion.utils import FolderOfImages |
|
|
|
|
|
|
|
|
web = server.web |
|
|
|
|
|
@server.PromptServer.instance.routes.get("/vhs/viewvideo") |
|
|
@server.PromptServer.instance.routes.get("/viewvideo") |
|
|
async def view_video(request): |
|
|
query = request.rel_url.query |
|
|
path_res = await resolve_path(query) |
|
|
if isinstance(path_res, web.Response): |
|
|
return path_res |
|
|
file, filename, output_dir = path_res |
|
|
|
|
|
if ffmpeg_path is None: |
|
|
|
|
|
if is_safe_path(output_dir, strict=True): |
|
|
return web.FileResponse(path=file) |
|
|
|
|
|
frame_rate = query.get('frame_rate', 8) |
|
|
if query.get('format', 'video') == "folder": |
|
|
os.makedirs(folder_paths.get_temp_directory(), exist_ok=True) |
|
|
concat_file = os.path.join(folder_paths.get_temp_directory(), "image_sequence_preview.txt") |
|
|
skip_first_images = int(query.get('skip_first_images', 0)) |
|
|
select_every_nth = int(query.get('select_every_nth', 1)) or 1 |
|
|
valid_images = get_sorted_dir_files_from_directory(file, skip_first_images, select_every_nth, FolderOfImages.IMG_EXTENSIONS) |
|
|
if len(valid_images) == 0: |
|
|
return web.Response(status=204) |
|
|
with open(concat_file, "w") as f: |
|
|
f.write("ffconcat version 1.0\n") |
|
|
for path in valid_images: |
|
|
f.write("file '" + os.path.abspath(path) + "'\n") |
|
|
f.write("duration 0.125\n") |
|
|
in_args = ["-safe", "0", "-i", concat_file] |
|
|
else: |
|
|
in_args = ["-i", file] |
|
|
if '%' in file: |
|
|
in_args = ['-framerate', str(frame_rate)] + in_args |
|
|
|
|
|
|
|
|
base_fps = 30 |
|
|
try: |
|
|
proc = await asyncio.create_subprocess_exec(ffmpeg_path, *in_args, '-t', |
|
|
'0','-f', 'null','-', stdout=subprocess.PIPE, |
|
|
stderr=subprocess.PIPE, stdin=subprocess.DEVNULL) |
|
|
_, res_stderr = await proc.communicate() |
|
|
|
|
|
match = re.search(': Video: (\\w+) .+, (\\d+) fps,', res_stderr.decode(*ENCODE_ARGS)) |
|
|
if match: |
|
|
base_fps = float(match.group(2)) |
|
|
if match.group(1) == 'vp9': |
|
|
|
|
|
in_args = ['-c:v', 'libvpx-vp9'] + in_args |
|
|
except subprocess.CalledProcessError as e: |
|
|
print("An error occurred in the ffmpeg prepass:\n" \ |
|
|
+ e.stderr.decode(*ENCODE_ARGS)) |
|
|
return web.Response(status=500) |
|
|
vfilters = [] |
|
|
target_rate = float(query.get('force_rate', 0)) or base_fps |
|
|
modified_rate = target_rate / (float(query.get('select_every_nth',1)) or 1) |
|
|
start_time = 0 |
|
|
if 'start_time' in query: |
|
|
start_time = float(query['start_time']) |
|
|
elif float(query.get('skip_first_frames', 0)) > 0: |
|
|
start_time = float(query.get('skip_first_frames'))/target_rate |
|
|
if start_time > 1/modified_rate: |
|
|
start_time += 1/modified_rate |
|
|
if start_time > 0: |
|
|
if start_time > 4: |
|
|
post_seek = ['-ss', '4'] |
|
|
pre_seek = ['-ss', str(start_time - 4)] |
|
|
else: |
|
|
post_seek = ['-ss', str(start_time)] |
|
|
pre_seek = [] |
|
|
else: |
|
|
pre_seek = [] |
|
|
post_seek = [] |
|
|
|
|
|
args = [ffmpeg_path, "-v", "error"] + pre_seek + in_args + post_seek |
|
|
if target_rate != 0: |
|
|
args += ['-r', str(modified_rate)] |
|
|
if query.get('force_size','Disabled') != "Disabled": |
|
|
size = query['force_size'].split('x') |
|
|
if size[0] == '?' or size[1] == '?': |
|
|
size[0] = "-2" if size[0] == '?' else f"'min({size[0]},iw)'" |
|
|
size[1] = "-2" if size[1] == '?' else f"'min({size[1]},ih)'" |
|
|
else: |
|
|
|
|
|
|
|
|
ar = float(size[0])/float(size[1]) |
|
|
vfilters.append(f"crop=if(gt({ar}\\,a)\\,iw\\,ih*{ar}):if(gt({ar}\\,a)\\,iw/{ar}\\,ih)") |
|
|
size = ':'.join(size) |
|
|
vfilters.append(f"scale={size}") |
|
|
if len(vfilters) > 0: |
|
|
args += ["-vf", ",".join(vfilters)] |
|
|
if float(query.get('frame_load_cap', 0)) > 0: |
|
|
args += ["-frames:v", query['frame_load_cap'].split('.')[0]] |
|
|
|
|
|
if query.get('deadline', 'realtime') == 'good': |
|
|
deadline = 'good' |
|
|
else: |
|
|
deadline = 'realtime' |
|
|
|
|
|
args += ['-c:v', 'libvpx-vp9','-deadline', deadline, '-cpu-used', '8', '-f', 'webm', '-'] |
|
|
|
|
|
try: |
|
|
proc = await asyncio.create_subprocess_exec(*args, stdout=subprocess.PIPE, |
|
|
stdin=subprocess.DEVNULL) |
|
|
try: |
|
|
resp = web.StreamResponse() |
|
|
resp.content_type = 'video/webm' |
|
|
resp.headers["Content-Disposition"] = f"filename=\"{filename}\"" |
|
|
await resp.prepare(request) |
|
|
while len(bytes_read := await proc.stdout.read(2**20)) != 0: |
|
|
await resp.write(bytes_read) |
|
|
|
|
|
await proc.wait() |
|
|
except (ConnectionResetError, ConnectionError) as e: |
|
|
proc.kill() |
|
|
except BrokenPipeError as e: |
|
|
pass |
|
|
return resp |
|
|
@server.PromptServer.instance.routes.get("/vhs/viewaudio") |
|
|
async def view_audio(request): |
|
|
query = request.rel_url.query |
|
|
path_res = await resolve_path(query) |
|
|
if isinstance(path_res, web.Response): |
|
|
return path_res |
|
|
file, filename, output_dir = path_res |
|
|
if ffmpeg_path is None: |
|
|
|
|
|
if is_safe_path(output_dir, strict=True): |
|
|
return web.FileResponse(path=file) |
|
|
|
|
|
in_args = ["-i", file] |
|
|
start_time = 0 |
|
|
if 'start_time' in query: |
|
|
start_time = float(query['start_time']) |
|
|
args = [ffmpeg_path, "-v", "error", '-vn'] + in_args + ['-ss', str(start_time)] |
|
|
if float(query.get('duration', 0)) > 0: |
|
|
args += ['-t', str(query['duration'])] |
|
|
if query.get('deadline', 'realtime') == 'good': |
|
|
deadline = 'good' |
|
|
else: |
|
|
deadline = 'realtime' |
|
|
|
|
|
args += ['-c:a', 'libopus','-deadline', deadline, '-cpu-used', '8', '-f', 'webm', '-'] |
|
|
try: |
|
|
proc = await asyncio.create_subprocess_exec(*args, stdout=subprocess.PIPE, |
|
|
stdin=subprocess.DEVNULL) |
|
|
try: |
|
|
resp = web.StreamResponse() |
|
|
resp.content_type = 'audio/webm' |
|
|
resp.headers["Content-Disposition"] = f"filename=\"{filename}\"" |
|
|
await resp.prepare(request) |
|
|
while len(bytes_read := await proc.stdout.read(2**20)) != 0: |
|
|
await resp.write(bytes_read) |
|
|
|
|
|
await proc.wait() |
|
|
except (ConnectionResetError, ConnectionError) as e: |
|
|
proc.kill() |
|
|
except BrokenPipeError as e: |
|
|
pass |
|
|
return resp |
|
|
|
|
|
query_cache = {} |
|
|
@server.PromptServer.instance.routes.get("/vhs/queryvideo") |
|
|
async def query_video(request): |
|
|
query = request.rel_url.query |
|
|
filepath = await resolve_path(query) |
|
|
|
|
|
if isinstance(filepath, web.Response): |
|
|
return filepath |
|
|
filepath = filepath[0] |
|
|
if filepath.endswith(".webp"): |
|
|
|
|
|
return web.json_response({}) |
|
|
if filepath in query_cache and query_cache[filepath][0] == os.stat(filepath).st_mtime: |
|
|
source = query_cache[filepath][1] |
|
|
else: |
|
|
source = {} |
|
|
try: |
|
|
with av.open(filepath) as cont: |
|
|
stream = cont.streams.video[0] |
|
|
source['fps'] = float(stream.average_rate) |
|
|
source['duration'] = float(cont.duration / av.time_base) |
|
|
|
|
|
if stream.codec_context.name == 'vp9': |
|
|
cc = av.Codec('libvpx-vp9', 'r').create() |
|
|
else: |
|
|
cc = stream |
|
|
def fit(): |
|
|
for packet in cont.demux(video=0): |
|
|
yield from cc.decode(packet) |
|
|
frame = next(fit()) |
|
|
|
|
|
source['size'] = [frame.width, frame.height] |
|
|
source['alpha'] = 'a' in frame.format.name |
|
|
source['frames'] = stream.metadata.get('NUMBER_OF_FRAMES', round(source['duration'] * source['fps'])) |
|
|
query_cache[filepath] = (os.stat(filepath).st_mtime, source) |
|
|
except Exception: |
|
|
pass |
|
|
if not 'frames' in source: |
|
|
return web.json_response({}) |
|
|
loaded = {} |
|
|
loaded['duration'] = source['duration'] |
|
|
loaded['duration'] -= float(query.get('start_time',0)) |
|
|
loaded['fps'] = float(query.get('force_rate', 0)) or source.get('fps',1) |
|
|
loaded['duration'] -= int(query.get('skip_first_frames', 0)) / loaded['fps'] |
|
|
loaded['fps'] /= int(query.get('select_every_nth', 1)) or 1 |
|
|
loaded['frames'] = round(loaded['duration'] * loaded['fps']) |
|
|
return web.json_response({'source': source, 'loaded': loaded}) |
|
|
|
|
|
async def resolve_path(query): |
|
|
if "filename" not in query: |
|
|
return web.Response(status=204) |
|
|
filename = query["filename"] |
|
|
|
|
|
|
|
|
if is_url(filename): |
|
|
file = await asyncio.to_thread(try_download_video, filename) or file |
|
|
filname, output_dir = os.path.split(file) |
|
|
return file, filename, output_dir |
|
|
else: |
|
|
filename, output_dir = folder_paths.annotated_filepath(filename) |
|
|
|
|
|
type = query.get("type", "output") |
|
|
if type == "path": |
|
|
|
|
|
|
|
|
output_dir, filename = os.path.split(strip_path(filename)) |
|
|
if output_dir is None: |
|
|
output_dir = folder_paths.get_directory_by_type(type) |
|
|
|
|
|
if output_dir is None: |
|
|
return web.Response(status=204) |
|
|
|
|
|
if not is_safe_path(output_dir): |
|
|
return web.Response(status=204) |
|
|
|
|
|
if "subfolder" in query: |
|
|
output_dir = os.path.join(output_dir, query["subfolder"]) |
|
|
|
|
|
filename = os.path.basename(filename) |
|
|
file = os.path.join(output_dir, filename) |
|
|
|
|
|
if not os.path.exists(file): |
|
|
return web.Response(status=204) |
|
|
if query.get('format', 'video') == 'folder': |
|
|
if not os.path.isdir(file): |
|
|
return web.Response(status=204) |
|
|
else: |
|
|
if not os.path.isfile(file) and not validate_sequence(file): |
|
|
return web.Response(status=204) |
|
|
return file, filename, output_dir |
|
|
|
|
|
@server.PromptServer.instance.routes.get("/vhs/getpath") |
|
|
@server.PromptServer.instance.routes.get("/getpath") |
|
|
async def get_path(request): |
|
|
query = request.rel_url.query |
|
|
if "path" not in query: |
|
|
return web.Response(status=204) |
|
|
|
|
|
path = os.path.abspath(strip_path(query["path"])) |
|
|
|
|
|
if not os.path.exists(path) or not is_safe_path(path): |
|
|
return web.json_response([]) |
|
|
|
|
|
|
|
|
valid_extensions = query.get("extensions") |
|
|
valid_items = [] |
|
|
for item in os.scandir(path): |
|
|
try: |
|
|
if item.is_dir(): |
|
|
valid_items.append(item.name + "/") |
|
|
continue |
|
|
if valid_extensions is None or item.name.split(".")[-1].lower() in valid_extensions: |
|
|
valid_items.append(item.name) |
|
|
except OSError: |
|
|
|
|
|
pass |
|
|
valid_items.sort(key=lambda f: os.stat(os.path.join(path,f)).st_mtime) |
|
|
return web.json_response(valid_items) |
|
|
|