camenduru commited on
Commit
1fb798e
1 Parent(s): b6e111c

Create worker_runpod.py

Browse files
Files changed (1) hide show
  1. worker_runpod.py +138 -0
worker_runpod.py ADDED
@@ -0,0 +1,138 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ import os, json, requests, random, time, runpod
2
+ from urllib.parse import urlsplit
3
+
4
+ import torch
5
+ from PIL import Image
6
+ import numpy as np
7
+
8
+ import asyncio
9
+ import execution
10
+ import server
11
+ loop = asyncio.new_event_loop()
12
+ asyncio.set_event_loop(loop)
13
+ server_instance = server.PromptServer(loop)
14
+ execution.PromptQueue(server)
15
+
16
+ from nodes import load_custom_node
17
+ from nodes import NODE_CLASS_MAPPINGS
18
+
19
+ load_custom_node("/content/ComfyUI/custom_nodes/ComfyUI-CogVideoXWrapper")
20
+ load_custom_node("/content/ComfyUI/custom_nodes/ComfyUI-VideoHelperSuite")
21
+ load_custom_node("/content/ComfyUI/custom_nodes/ComfyUI-KJNodes")
22
+
23
+ LoadImage = NODE_CLASS_MAPPINGS["LoadImage"]()
24
+ ImageResizeKJ = NODE_CLASS_MAPPINGS["ImageResizeKJ"]()
25
+ CogVideoImageEncode = NODE_CLASS_MAPPINGS["CogVideoImageEncode"]()
26
+ CogVideoLoraSelect = NODE_CLASS_MAPPINGS["CogVideoLoraSelect"]()
27
+ DownloadAndLoadCogVideoModel = NODE_CLASS_MAPPINGS["DownloadAndLoadCogVideoModel"]()
28
+ CogVideoTextEncode = NODE_CLASS_MAPPINGS["CogVideoTextEncode"]()
29
+ CLIPLoader = NODE_CLASS_MAPPINGS["CLIPLoader"]()
30
+ CogVideoSampler = NODE_CLASS_MAPPINGS["CogVideoSampler"]()
31
+ CogVideoDecode = NODE_CLASS_MAPPINGS["CogVideoDecode"]()
32
+ VHS_VideoCombine = NODE_CLASS_MAPPINGS["VHS_VideoCombine"]()
33
+
34
+ with torch.inference_mode():
35
+ lora = CogVideoLoraSelect.getlorapath("orbit_left_lora_weights.safetensors", 1.0, fuse_lora=True)[0]
36
+ pipeline = DownloadAndLoadCogVideoModel.loadmodel("THUDM/CogVideoX-5b-I2V", "bf16", fp8_transformer="disabled", compile="disabled", enable_sequential_cpu_offload=False, lora=lora)[0]
37
+ clip = CLIPLoader.load_clip("t5xxl_fp16.safetensors", type="sd3")[0]
38
+
39
+ def download_file(url, save_dir, file_name):
40
+ os.makedirs(save_dir, exist_ok=True)
41
+ original_file_name = url.split('/')[-1]
42
+ _, original_file_extension = os.path.splitext(original_file_name)
43
+ file_path = os.path.join(save_dir, file_name + original_file_extension)
44
+ response = requests.get(url)
45
+ response.raise_for_status()
46
+ with open(file_path, 'wb') as file:
47
+ file.write(response.content)
48
+ return file_path
49
+
50
+ @torch.inference_mode()
51
+ def generate(input):
52
+ values = input["input"]
53
+
54
+ input_image=values['input_image_check']
55
+ input_image=download_file(url=input_image, save_dir='/content/ComfyUI/input', file_name='input_image')
56
+ prompt = values['prompt']
57
+ negative_prompt = values['negative_prompt']
58
+ seed = values['seed']
59
+ steps = values['steps']
60
+ cfg = values['cfg']
61
+
62
+ if seed == 0:
63
+ random.seed(int(time.time()))
64
+ seed = random.randint(0, 18446744073709551615)
65
+
66
+ positive = CogVideoTextEncode.process(clip, prompt, strength=1.0, force_offload=True)[0]
67
+ negative = CogVideoTextEncode.process(clip, negative_prompt, strength=1.0, force_offload=True)[0]
68
+
69
+ image, _ = LoadImage.load_image(input_image)
70
+ image = ImageResizeKJ.resize(image, width=720, height=480, keep_proportion=False, upscale_method="lanczos", divisible_by=16, crop="center")[0]
71
+ image_cond_latents = CogVideoImageEncode.encode(pipeline, image, chunk_size=16, enable_tiling=True)[0]
72
+ samples = CogVideoSampler.process(pipeline, positive, negative, steps, cfg, seed, height=480, width=720, num_frames=49, scheduler="CogVideoXDPMScheduler", denoise_strength=1.0, image_cond_latents=image_cond_latents)
73
+ frames = CogVideoDecode.decode(samples[0], samples[1], enable_vae_tiling=True, tile_sample_min_height=240, tile_sample_min_width=360, tile_overlap_factor_height=0.2, tile_overlap_factor_width=0.2, auto_tile_size=True)[0]
74
+
75
+ out_video = VHS_VideoCombine.combine_video(images=frames, frame_rate=8, loop_count=0, filename_prefix="CogVideoX-I2V", format="video/h264-mp4", save_output=True)
76
+ source = out_video["result"][0][1][1]
77
+ destination = f"/content/ComfyUI/output/cogvideox-5b-i2v-dimensionx-{seed}-tost.mp4"
78
+ shutil.move(source, destination)
79
+
80
+ result = f"/content/ComfyUI/output/cogvideox-5b-i2v-dimensionx-{seed}-tost.mp4"
81
+ try:
82
+ notify_uri = values['notify_uri']
83
+ del values['notify_uri']
84
+ notify_token = values['notify_token']
85
+ del values['notify_token']
86
+ discord_id = values['discord_id']
87
+ del values['discord_id']
88
+ if(discord_id == "discord_id"):
89
+ discord_id = os.getenv('com_camenduru_discord_id')
90
+ discord_channel = values['discord_channel']
91
+ del values['discord_channel']
92
+ if(discord_channel == "discord_channel"):
93
+ discord_channel = os.getenv('com_camenduru_discord_channel')
94
+ discord_token = values['discord_token']
95
+ del values['discord_token']
96
+ if(discord_token == "discord_token"):
97
+ discord_token = os.getenv('com_camenduru_discord_token')
98
+ job_id = values['job_id']
99
+ del values['job_id']
100
+ default_filename = os.path.basename(result)
101
+ with open(result, "rb") as file:
102
+ files = {default_filename: file.read()}
103
+ payload = {"content": f"{json.dumps(values)} <@{discord_id}>"}
104
+ response = requests.post(
105
+ f"https://discord.com/api/v9/channels/{discord_channel}/messages",
106
+ data=payload,
107
+ headers={"Authorization": f"Bot {discord_token}"},
108
+ files=files
109
+ )
110
+ response.raise_for_status()
111
+ result_url = response.json()['attachments'][0]['url']
112
+ notify_payload = {"jobId": job_id, "result": result_url, "status": "DONE"}
113
+ web_notify_uri = os.getenv('com_camenduru_web_notify_uri')
114
+ web_notify_token = os.getenv('com_camenduru_web_notify_token')
115
+ if(notify_uri == "notify_uri"):
116
+ requests.post(web_notify_uri, data=json.dumps(notify_payload), headers={'Content-Type': 'application/json', "Authorization": web_notify_token})
117
+ else:
118
+ requests.post(web_notify_uri, data=json.dumps(notify_payload), headers={'Content-Type': 'application/json', "Authorization": web_notify_token})
119
+ requests.post(notify_uri, data=json.dumps(notify_payload), headers={'Content-Type': 'application/json', "Authorization": notify_token})
120
+ return {"jobId": job_id, "result": result_url, "status": "DONE"}
121
+ except Exception as e:
122
+ error_payload = {"jobId": job_id, "status": "FAILED"}
123
+ try:
124
+ if(notify_uri == "notify_uri"):
125
+ requests.post(web_notify_uri, data=json.dumps(error_payload), headers={'Content-Type': 'application/json', "Authorization": web_notify_token})
126
+ else:
127
+ requests.post(web_notify_uri, data=json.dumps(error_payload), headers={'Content-Type': 'application/json', "Authorization": web_notify_token})
128
+ requests.post(notify_uri, data=json.dumps(error_payload), headers={'Content-Type': 'application/json', "Authorization": notify_token})
129
+ except:
130
+ pass
131
+ return {"jobId": job_id, "result": f"FAILED: {str(e)}", "status": "FAILED"}
132
+ finally:
133
+ if os.path.exists(result):
134
+ os.remove(result)
135
+ if os.path.exists(input_image):
136
+ os.remove(input_image)
137
+
138
+ runpod.serverless.start({"handler": generate})