#!/usr/bin/env python # -*- coding: UTF-8 -*- ''' @Project :EchoMimic @File :audio2vid.py @Author :juzhen.czy @Date :2024/3/4 17:43 ''' import argparse import os import random import platform import subprocess from datetime import datetime from pathlib import Path import cv2 import numpy as np import torch from diffusers import AutoencoderKL, DDIMScheduler from omegaconf import OmegaConf from PIL import Image from src.models.unet_2d_condition import UNet2DConditionModel from src.models.unet_3d_echo import EchoUNet3DConditionModel from src.models.whisper.audio2feature import load_audio_model from src.pipelines.pipeline_echo_mimic import Audio2VideoPipeline from src.utils.util import save_videos_grid, crop_and_pad from src.models.face_locator import FaceLocator from moviepy.editor import VideoFileClip, AudioFileClip from facenet_pytorch import MTCNN ffmpeg_path = os.getenv('FFMPEG_PATH') if ffmpeg_path is None and platform.system() in ['Linux', 'Darwin']: try: result = subprocess.run(['which', 'ffmpeg'], capture_output=True, text=True) if result.returncode == 0: ffmpeg_path = result.stdout.strip() print(f"FFmpeg is installed at: {ffmpeg_path}") else: print("FFmpeg is not installed. Please download ffmpeg-static and export to FFMPEG_PATH.") print("For example: export FFMPEG_PATH=/musetalk/ffmpeg-4.4-amd64-static") except Exception as e: pass if ffmpeg_path is not None and ffmpeg_path not in os.getenv('PATH'): print("Adding FFMPEG_PATH to PATH") os.environ["PATH"] = f"{ffmpeg_path}:{os.environ['PATH']}" def parse_args(): parser = argparse.ArgumentParser() parser.add_argument("--config", type=str, default="./configs/prompts/animation.yaml") parser.add_argument("-W", type=int, default=512) parser.add_argument("-H", type=int, default=512) parser.add_argument("-L", type=int, default=1200) parser.add_argument("--seed", type=int, default=420) parser.add_argument("--facemusk_dilation_ratio", type=float, default=0.1) parser.add_argument("--facecrop_dilation_ratio", type=float, default=0.5) parser.add_argument("--context_frames", type=int, default=12) parser.add_argument("--context_overlap", type=int, default=3) parser.add_argument("--cfg", type=float, default=2.5) parser.add_argument("--steps", type=int, default=30) parser.add_argument("--sample_rate", type=int, default=16000) parser.add_argument("--fps", type=int, default=24) parser.add_argument("--device", type=str, default="cuda") args = parser.parse_args() return args def select_face(det_bboxes, probs): ## max face from faces that the prob is above 0.8 ## box: xyxy if det_bboxes is None or probs is None: return None filtered_bboxes = [] for bbox_i in range(len(det_bboxes)): if probs[bbox_i] > 0.8: filtered_bboxes.append(det_bboxes[bbox_i]) if len(filtered_bboxes) == 0: return None sorted_bboxes = sorted(filtered_bboxes, key=lambda x:(x[3]-x[1]) * (x[2] - x[0]), reverse=True) return sorted_bboxes[0] def main(): args = parse_args() config = OmegaConf.load(args.config) if config.weight_dtype == "fp16": weight_dtype = torch.float16 else: weight_dtype = torch.float32 device = args.device if device.__contains__("cuda") and not torch.cuda.is_available(): device = "cpu" inference_config_path = config.inference_config infer_config = OmegaConf.load(inference_config_path) ############# model_init started ############# ## vae init vae = AutoencoderKL.from_pretrained( config.pretrained_vae_path, ).to("cuda", dtype=weight_dtype) ## reference net init reference_unet = UNet2DConditionModel.from_pretrained( config.pretrained_base_model_path, subfolder="unet", ).to(dtype=weight_dtype, device=device) reference_unet.load_state_dict( torch.load(config.reference_unet_path, map_location="cpu"), ) ## denoising net init if os.path.exists(config.motion_module_path): ### stage1 + stage2 denoising_unet = EchoUNet3DConditionModel.from_pretrained_2d( config.pretrained_base_model_path, config.motion_module_path, subfolder="unet", unet_additional_kwargs=infer_config.unet_additional_kwargs, ).to(dtype=weight_dtype, device=device) else: ### only stage1 denoising_unet = EchoUNet3DConditionModel.from_pretrained_2d( config.pretrained_base_model_path, "", subfolder="unet", unet_additional_kwargs={ "use_motion_module": False, "unet_use_temporal_attention": False, "cross_attention_dim": infer_config.unet_additional_kwargs.cross_attention_dim } ).to(dtype=weight_dtype, device=device) denoising_unet.load_state_dict( torch.load(config.denoising_unet_path, map_location="cpu"), strict=False ) ## face locator init face_locator = FaceLocator(320, conditioning_channels=1, block_out_channels=(16, 32, 96, 256)).to( dtype=weight_dtype, device="cuda" ) face_locator.load_state_dict(torch.load(config.face_locator_path)) ### load audio processor params audio_processor = load_audio_model(model_path=config.audio_model_path, device=device) ### load face detector params face_detector = MTCNN(image_size=320, margin=0, min_face_size=20, thresholds=[0.6, 0.7, 0.7], factor=0.709, post_process=True, device=device) ############# model_init finished ############# width, height = args.W, args.H sched_kwargs = OmegaConf.to_container(infer_config.noise_scheduler_kwargs) scheduler = DDIMScheduler(**sched_kwargs) pipe = Audio2VideoPipeline( vae=vae, reference_unet=reference_unet, denoising_unet=denoising_unet, audio_guider=audio_processor, face_locator=face_locator, scheduler=scheduler, ) pipe = pipe.to("cuda", dtype=weight_dtype) date_str = datetime.now().strftime("%Y%m%d") time_str = datetime.now().strftime("%H%M") save_dir_name = f"{time_str}--seed_{args.seed}-{args.W}x{args.H}" save_dir = Path(f"output/{date_str}/{save_dir_name}") save_dir.mkdir(exist_ok=True, parents=True) for ref_image_path in config["test_cases"].keys(): for audio_path in config["test_cases"][ref_image_path]: if args.seed is not None and args.seed > -1: generator = torch.manual_seed(args.seed) else: generator = torch.manual_seed(random.randint(100, 1000000)) ref_name = Path(ref_image_path).stem audio_name = Path(audio_path).stem final_fps = args.fps #### face musk prepare face_img = cv2.imread(ref_image_path) face_mask = np.zeros((face_img.shape[0], face_img.shape[1])).astype('uint8') det_bboxes, probs = face_detector.detect(face_img) select_bbox = select_face(det_bboxes, probs) if select_bbox is None: face_mask[:, :] = 255 else: xyxy = select_bbox[:4] xyxy = np.round(xyxy).astype('int') rb, re, cb, ce = xyxy[1], xyxy[3], xyxy[0], xyxy[2] r_pad = int((re - rb) * args.facemusk_dilation_ratio) c_pad = int((ce - cb) * args.facemusk_dilation_ratio) face_mask[rb - r_pad : re + r_pad, cb - c_pad : ce + c_pad] = 255 #### face crop r_pad_crop = int((re - rb) * args.facecrop_dilation_ratio) c_pad_crop = int((ce - cb) * args.facecrop_dilation_ratio) crop_rect = [max(0, cb - c_pad_crop), max(0, rb - r_pad_crop), min(ce + c_pad_crop, face_img.shape[1]), min(re + c_pad_crop, face_img.shape[0])] print(crop_rect) face_img = crop_and_pad(face_img, crop_rect) face_mask = crop_and_pad(face_mask, crop_rect) face_img = cv2.resize(face_img, (args.W, args.H)) face_mask = cv2.resize(face_mask, (args.W, args.H)) ref_image_pil = Image.fromarray(face_img[:, :, [2, 1, 0]]) face_mask_tensor = torch.Tensor(face_mask).to(dtype=weight_dtype, device="cuda").unsqueeze(0).unsqueeze(0).unsqueeze(0) / 255.0 video = pipe( ref_image_pil, audio_path, face_mask_tensor, width, height, args.L, args.steps, args.cfg, generator=generator, audio_sample_rate=args.sample_rate, context_frames=args.context_frames, fps=final_fps, context_overlap=args.context_overlap ).videos video = video save_videos_grid( video, f"{save_dir}/{ref_name}_{audio_name}_{args.H}x{args.W}_{int(args.cfg)}_{time_str}.mp4", n_rows=1, fps=final_fps, ) video_clip = VideoFileClip(f"{save_dir}/{ref_name}_{audio_name}_{args.H}x{args.W}_{int(args.cfg)}_{time_str}.mp4") audio_clip = AudioFileClip(audio_path) video_clip = video_clip.set_audio(audio_clip) video_clip.write_videofile(f"{save_dir}/{ref_name}_{audio_name}_{args.H}x{args.W}_{int(args.cfg)}_{time_str}_withaudio.mp4", codec="libx264", audio_codec="aac") print(f"{save_dir}/{ref_name}_{audio_name}_{args.H}x{args.W}_{int(args.cfg)}_{time_str}_withaudio.mp4") if __name__ == "__main__": main()