File size: 9,567 Bytes
ad5a806 027e8a9 97515ed b345f1b 027e8a9 ad6ae4a 027e8a9 b345f1b 027e8a9 b345f1b 027e8a9 b345f1b 003efa5 2aabdcd 003efa5 7c8a760 003efa5 b345f1b 39ff4a4 a550e5b 63c362f 2c12515 fa29376 2c12515 b345f1b 91c5ebc b345f1b 91c5ebc 042d40b 53b4fd0 042d40b 91c5ebc 53b4fd0 91c5ebc b345f1b 91c5ebc 97515ed b345f1b 53b4fd0 9da309a 042d40b 53b4fd0 91c5ebc b345f1b d682a07 ad6ae4a b345f1b 933471e 97515ed b345f1b ad5a806 3b6a87c b345f1b 027e8a9 b345f1b 1b6024a b345f1b 027e8a9 b345f1b fa29376 b345f1b 24987c8 ad6ae4a fa29376 ad6ae4a b345f1b 70f5d8a b345f1b fa29376 933471e 70f5d8a 933471e 027e8a9 7c60508 24987c8 7c60508 ae0f617 7c60508 027e8a9 fa29376 4de0933 fa29376 027e8a9 ad6ae4a 2c12515 4de0933 59a74a9 cd5d77e 2c12515 52be6ba 57c5811 3c3adba 52be6ba 2c12515 ad6ae4a a550e5b 63c362f 2c12515 63c362f 027e8a9 63c362f 027e8a9 bbf1141 |
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 |
import spaces
import gradio as gr
from gradio_client import Client
moondream_client = Client("https://vikhyatk-moondream1.hf.space/")
salmmon_client = Client("fffiloni/SALMONN-7B-gradio")
import cv2
from moviepy.editor import *
# 1. extract and store 1 image every 24 images from video input
# 2. extract audio
# 3. for each image from extracted_images, get caption from caption model and concatenate into list
# 4. for audio, ask audio questioning model to describe sound/scene
# 5. give all to LLM, and ask it to resume, according to image caption list combined to audio caption
import re
import torch
from transformers import pipeline
zephyr_model = "HuggingFaceH4/zephyr-7b-beta"
pipe = pipeline("text-generation", model=zephyr_model, torch_dtype=torch.bfloat16, device_map="auto")
standard_sys = f"""
You will be provided a list of visual details observed at regular intervals, along with an audio description. These pieces of information originate from a single video. The visual details are extracted from the video at fixed time intervals and represent consecutive frames. Typically, the video consists of a brief sequence showing one or more subjects...
Please note that the following list of image descriptions (visual details) was obtained by extracting individual frames from a continuous video featuring one or more subjects. Depending on the case, all depicted individuals may correspond to the same person(s), with minor variations due to changes in lighting, angle, and facial expressions over time. Regardless, assume temporal continuity among the frames unless otherwise specified.
Audio events are actually the entire scene description based only on the audio of the video. Your job is to integrate these multimodal inputs intelligently and provide a very short resume about what is happening in the origin video. Provide a succinct overview of what you understood.
"""
def trim_video(input_path, max_duration=10):
if input_path is not None:
video_clip = VideoFileClip(input_path)
output_path = "video_cut_10.mp4"
if video_clip.duration > max_duration:
trimmed_clip = video_clip.subclip(0, max_duration)
trimmed_clip.write_videofile(output_path, audio_codec='aac')
return output_path
else:
return input_path
else :
return None
def extract_frames(video_in, output_format='.jpg'):
# Adjust interval to video length
video_clip = VideoFileClip(video_in)
if video_clip.duration <= 5:
interval = 6
else :
interval = 24
"""Extract frames from a video at a specified interval and store them in a list.
Args:
- video_in: string or path-like object pointing to the video file
- interval: integer specifying how many frames apart to extract images (default: 5)
- output_format: string indicating desired format for saved images (default: '.jpg')
Returns:
A list of strings containing paths to saved images.
"""
# Initialize variables
vidcap = cv2.VideoCapture(video_in)
frames = []
count = 0
# Loop through frames until there are no more
while True:
success, image = vidcap.read()
# Check if successful read and not past end of video
if success:
#print('Read a new frame:', success)
# Save current frame if it meets criteria
if count % interval == 0:
filename = f'frame_{count // interval}{output_format}'
frames.append(filename)
cv2.imwrite(filename, image)
print(f'Saved {filename}')
# Increment counter
count += 1
# Break out of loop when done reading frames
else:
break
# Close video capture
vidcap.release()
print('Done extracting frames!')
return frames
'''
from transformers import AutoModelForCausalLM, AutoTokenizer
from PIL import Image
model_id = "vikhyatk/moondream2"
revision = "2024-03-06"
model = AutoModelForCausalLM.from_pretrained(
model_id, trust_remote_code=True, revision=revision
)
tokenizer = AutoTokenizer.from_pretrained(model_id, revision=revision)
'''
#@spaces.GPU()
def process_image(image_in):
result = moondream_client.predict(
image_in, # filepath in 'image' Image component
"Describe precisely the image in one sentence.", # str in 'Question' Textbox component
api_name="/answer_question"
#api_name="/predict"
)
print(result)
return result
'''
image = Image.open(image_in)
enc_image = model.encode_image(image)
result = model.answer_question(enc_image, "Describe the image in one sentence.", tokenizer)
print(result)
return result
'''
def extract_audio(video_path):
video_clip = VideoFileClip(video_path)
# Check if the video has audio
if video_clip.audio is not None:
audio_clip = video_clip.audio
audio_clip.write_audiofile("output_audio.mp3")
return "output_audio.mp3"
else:
print("The video does not have any audio.")
return None
def get_salmonn(audio_in):
salmonn_prompt = "Please describe the audio"
result = salmmon_client.predict(
audio_in, # filepath in 'Audio' Audio component
salmonn_prompt, # str in 'User question' Textbox component
4, # float (numeric value between 1 and 10) in 'beam search numbers' Slider component
1, # float (numeric value between 0.8 and 2.0) in 'temperature' Slider component
0.9, # float (numeric value between 0.1 and 1.0) in 'top p' Slider component
api_name="/gradio_answer"
)
print(result)
return result
@spaces.GPU()
def llm_process(user_prompt):
agent_maker_sys = standard_sys
instruction = f"""
<|system|>
{agent_maker_sys}</s>
<|user|>
"""
prompt = f"{instruction.strip()}\n{user_prompt}</s>"
outputs = pipe(prompt, max_new_tokens=256, do_sample=True, temperature=0.7, top_k=50, top_p=0.95)
pattern = r'\<\|system\|\>(.*?)\<\|assistant\|\>'
cleaned_text = re.sub(pattern, '', outputs[0]["generated_text"], flags=re.DOTALL)
print(f"SUGGESTED video description: {cleaned_text}")
return cleaned_text.lstrip("\n")
def infer(video_in):
# Extract frames from a video
gr.Info("Extracting frames...")
frame_files = extract_frames(video_in)
# Process each extracted frame and collect results in a list
gr.Info("Captioning frames ...")
processed_texts = []
for frame_file in frame_files:
text = process_image(frame_file)
processed_texts.append(text)
print(processed_texts)
# Convert processed_texts list to a string list with line breaks
string_list = '\n'.join(processed_texts)
# Extract audio from video
extracted_audio = extract_audio(video_in)
if extracted_audio is not None :
print(extracted_audio)
# Get description of audio content
gr.Info("Getting audio description from extracted sound ...")
audio_content_described = get_salmonn(extracted_audio)
else :
audio_content_described = "Video has no sound."
# Assemble captions
formatted_captions = f"""
### Visual events:\n{string_list}\n ### Audio events:\n{audio_content_described}
"""
print(formatted_captions)
# Send formatted captions to LLM
gr.Info("Try to provide a video understanding with provided elements ...")
video_description_from_llm = llm_process(formatted_captions)
return video_description_from_llm
css = """
div#col-container{
margin: 0 auto;
max-width: 1280px;
}
div#video-text textarea {
font-size: 20px;
line-height: 1.2em;
font-weight: 600;
}
"""
with gr.Blocks(css=css) as demo :
with gr.Column(elem_id="col-container"):
gr.HTML("""
<h2 style="text-align: center;">Soft Video Understanding</h2>
<p style="text-align: center;">
An experiment to try to achieve what i call "soft video understanding" with open-source available models. <br />
We use moondream1 to caption extracted frames, salmonn to analyze extracted audio, then give visual and audio details to Zephyr which is instructed to resume what it understood.<br />
Instructions prompt is available for further discussion with the Community. <br />
Note that audio is crucial for better overall vision. Video longer than 10 seconds will be cut.
</p>
""")
with gr.Row():
with gr.Column():
video_in = gr.Video(label="Video input")
with gr.Accordion("System Instructions (for your curiosity)", open=False):
system_instruction = gr.Markdown(
value = standard_sys
)
gr.Examples(
examples = ["examples/train.mp4", "examples/puppies.mp4", "examples/turtle.mp4"],
inputs = [video_in]
)
with gr.Column():
video_cut = gr.Video(label="Video cut to 10 seconds", interactive=False)
submit_btn = gr.Button("Submit")
video_description = gr.Textbox(label="Video description", elem_id="video-text")
video_in.change(
fn = trim_video,
inputs = [video_in],
outputs = [video_cut],
queue = False
)
submit_btn.click(
fn = infer,
inputs = [video_cut],
outputs = [video_description]
)
demo.queue(max_size=10).launch(show_error=True) |