diva-audio-chat / app.py
Helw150
Spinning DiVA
0d5fe86
raw
history blame
8.63 kB
import io
import os
import time
import traceback
from dataclasses import dataclass, field
import gradio as gr
import librosa
import numpy as np
import pvorca
import soundfile as sf
import spaces
import torch
import xxhash
from datasets import Audio
from transformers import AutoModel
from transformers.modeling_outputs import CausalLMOutputWithPast
orca = pvorca.create(
access_key=os.environ.get("ORCA_KEY"),
model_path="./static/orca_params_masculine.pv",
)
LOADER_STR = "♫♪.ılılıll|̲̅̅●̲̅̅|̲̅̅=̲̅̅|̲̅̅●̲̅̅|llılılı.♫♪loading♫♪.ılılıll|̲̅̅●̲̅̅|̲̅̅=̲̅̅|̲̅̅●̲̅̅|llılılı.♫♪loading♫♪.ılılıll|̲̅̅●̲̅̅|̲̅̅=̲̅̅|̲̅̅●̲̅̅|llılılı.♫♪♫"
if gr.NO_RELOAD:
diva_model = AutoModel.from_pretrained(
"WillHeld/DiVA-llama-3-v0-8b", trust_remote_code=True
)
resampler = Audio(sampling_rate=16_000)
@spaces.GPU
@torch.no_grad
def diva_audio(audio_input, do_sample=False, temperature=0.001, prev_outs=None):
sr, y = audio_input
x = xxhash.xxh32(bytes(y)).hexdigest()
y = y.astype(np.float32)
y /= np.max(np.abs(y))
a = resampler.decode_example(
resampler.encode_example({"array": y, "sampling_rate": sr})
)
yield from diva_model.generate_stream(
a["array"],
(
"Your name is DiVA, which stands for Distilled Voice Assistant. You were trained with early-fusion training to merge OpenAI's Whisper and Meta AI's Llama 3 8B to provide end-to-end voice processing. You should give brief and helpful answers, in a conversational style. The user is talking to you with their voice and you are responding with text."
if prev_outs == None
else None
),
do_sample=do_sample,
max_new_tokens=256,
init_outputs=prev_outs,
return_outputs=True,
)
@dataclass
class AppState:
conversation: list = field(default_factory=list)
stopped: bool = False
model_outs: any = None
def process_audio(audio: tuple, state: AppState):
return audio, state
@spaces.GPU(duration=40, progress=gr.Progress(track_tqdm=True))
def response(state: AppState, audio: tuple):
if not audio:
return AppState()
file_name = f"/tmp/{xxhash.xxh32(bytes(audio[1])).hexdigest()}.wav"
sf.write(file_name, audio[1], audio[0], format="wav")
state.conversation.append(
{"role": "user", "content": {"path": file_name, "mime_type": "audio/wav"}}
)
gr.Warning(
"The first response might take a second to generate as DiVA is loaded from Disk to the ZeroGPU!"
)
state.conversation.append(
{
"role": "assistant",
"content": LOADER_STR,
}
)
yield state, state.conversation, None
if spaces.config.Config.zero_gpu:
if state.model_outs is not None:
state.model_outs = tuple(
tuple(torch.tensor(vec).cuda() for vec in tup)
for tup in state.model_outs
)
causal_outs = (
CausalLMOutputWithPast(past_key_values=state.model_outs)
if state.model_outs
else None
)
else:
causal_outs = state.model_outs
state.model_outs = None
prev_outs = causal_outs
stream = orca.stream_open()
for resp, outs in diva_audio(
(audio[0], audio[1]),
prev_outs=(prev_outs if prev_outs is not None else None),
):
prev_resp = state.conversation[-1]["content"]
if prev_resp == LOADER_STR:
prev_resp = ""
state.conversation[-1]["content"] = resp
pcm = stream.synthesize(resp[len(prev_resp) :])
audio_chunk = None
if pcm is not None:
mp3_io = io.BytesIO()
sf.write(
mp3_io, np.asarray(pcm).astype(np.int16), orca.sample_rate, format="mp3"
)
audio_chunk = mp3_io.getvalue()
mp3_io.close()
yield state, state.conversation, audio_chunk
del outs.logits
del outs.hidden_states
if spaces.config.Config.zero_gpu:
outs = tuple(
tuple(vec.cpu().numpy() for vec in tup) for tup in outs.past_key_values
)
audio_chunk = None
pcm = stream.flush()
if pcm is not None:
audio_chunk = np.asarray(pcm).tobytes()
mp3_io = io.BytesIO()
sf.write(
mp3_io, np.asarray(pcm).astype(np.int16), orca.sample_rate, format="mp3"
)
audio_chunk = mp3_io.getvalue()
mp3_io.close()
stream.close()
yield (
AppState(conversation=state.conversation, model_outs=outs),
state.conversation,
audio_chunk,
)
def start_recording_user(state: AppState):
return None
theme = gr.themes.Soft(
primary_hue=gr.themes.Color(
c100="#82000019",
c200="#82000033",
c300="#8200004c",
c400="#82000066",
c50="#8200007f",
c500="#8200007f",
c600="#82000099",
c700="#820000b2",
c800="#820000cc",
c900="#820000e5",
c950="#820000f2",
),
secondary_hue="rose",
neutral_hue="stone",
)
js = """
async function main() {
const script1 = document.createElement("script");
script1.src = "https://cdn.jsdelivr.net/npm/onnxruntime-web@1.14.0/dist/ort.js";
document.head.appendChild(script1)
const script2 = document.createElement("script");
script2.onload = async () => {
console.log("vad loaded") ;
var record = document.querySelector('.record-button');
record.textContent = "Just Start Talking!"
record.style = "width: 11vw"
const myvad = await vad.MicVAD.new({
onSpeechStart: () => {
var record = document.querySelector('.record-button');
if (record != null) {
console.log(record);
record.click();
var logo = document.querySelector('.spinning-logo');
logo.style = "animation: rotation 10s infinite linear; height: 10vw; margin: auto;"
}
},
onSpeechEnd: (audio) => {
var stop = document.querySelector('.stop-button');
if (stop != null) {
console.log(stop);
stop.click();
var logo = document.querySelector('.spinning-logo');
logo.style = "animation: rotation 1s infinite linear; height: 10vw; margin: auto;"
}
}
})
myvad.start()
}
script2.src = "https://cdn.jsdelivr.net/npm/@ricky0123/vad-web@0.0.7/dist/bundle.min.js";
script1.onload = () => {
console.log("onnx loaded")
document.head.appendChild(script2)
};
}
"""
js_reset = """
() => {
var record = document.querySelector('.record-button');
record.textContent = "Just Start Talking!"
record.style = "width: 11vw"
var logo = document.querySelector('.spinning-logo');
logo.style = "height: 10vw; margin: auto;"
}
"""
css = """
@keyframes rotation {
from {
transform: rotate(0deg);
}
to {
transform: rotate(360deg);
}
}
"""
with gr.Blocks(theme=theme, js=js, css=css) as demo:
with gr.Row():
with gr.Column(scale=9):
gr.HTML(
"""
<img class="spinning-logo" src="https://diva-audio.github.io/static/DiVA-logo.png" alt="A Cute Small Robot with DiVA Written on It's Chest" style="height: 10vw; margin: auto;">
"""
)
with gr.Column(scale=1):
input_audio = gr.Audio(
label="Input Audio",
sources=["microphone"],
type="numpy",
streaming=False,
waveform_options=gr.WaveformOptions(show_recording_waveform=False),
)
with gr.Row():
chatbot = gr.Chatbot(label="Conversation", type="messages")
with gr.Row(max_height="50vh"):
output_audio = gr.Audio(
label="Output Audio",
streaming=True,
autoplay=True,
visible=False,
)
state = gr.State(value=AppState())
stream = input_audio.start_recording(
process_audio,
[input_audio, state],
[input_audio, state],
)
respond = input_audio.stop_recording(
response, [state, input_audio], [state, chatbot, output_audio]
)
restart = output_audio.stop(start_recording_user, [state], [input_audio]).then(
lambda state: state, state, state, js=js_reset
)
cancel = gr.Button("Restart Conversation", variant="stop")
cancel.click(
lambda: (AppState(), gr.Audio(recording=False)),
None,
[state, input_audio],
cancels=[respond, restart],
)
if __name__ == "__main__":
demo.launch()