LocalScribe1 / app.py
KG0101's picture
Add transcription transfer to editing text box
5ad7cdb verified
import spaces
import torch
import gradio as gr
from transformers import pipeline, AutoModelForCausalLM, AutoTokenizer, TextIteratorStreamer
from threading import Thread
from typing import Iterator
import os
MODEL_NAME = "openai/whisper-large-v3-turbo"
BATCH_SIZE = 8
FILE_LIMIT_MB = 5000
MAX_INPUT_TOKEN_LENGTH = int(os.getenv("MAX_INPUT_TOKEN_LENGTH", "4096"))
device = 0 if torch.cuda.is_available() else "cpu"
# Initialize the LLM
if torch.cuda.is_available():
llm_model_id = "chuanli11/Llama-3.2-3B-Instruct-uncensored"
llm = AutoModelForCausalLM.from_pretrained(llm_model_id, torch_dtype=torch.float16, device_map="auto")
tokenizer = AutoTokenizer.from_pretrained(llm_model_id)
tokenizer.use_default_system_prompt = False
# Initialize the transcription pipeline
pipe = pipeline(
task="automatic-speech-recognition",
model=MODEL_NAME,
chunk_length_s=30,
device=device,
)
# Function to transcribe audio inputs
@spaces.GPU
def transcribe(inputs, task):
if inputs is None:
raise gr.Error("No audio file submitted! Please upload or record an audio file before submitting your request.")
text = pipe(inputs, batch_size=BATCH_SIZE, generate_kwargs={"task": task}, return_timestamps=True)["text"]
return text
# Function to generate SOAP notes using LLM
@spaces.GPU
def generate_soap(
transcribed_text: str,
system_prompt: str = "You are a world class clinical assistant.",
max_new_tokens: int = 4098,
temperature: float = 0.6,
top_p: float = 0.9,
top_k: int = 50,
repetition_penalty: float = 1.2,
) -> Iterator[str]:
task_prompt = """
Convert the following transcribed conversation into a clinical SOAP note.
The text includes dialogue between a physician and a patient. Please clearly distinguish between the physician's and the patient's statements.
Extract and organize the information into the relevant sections of a SOAP note:
- Subjective (symptoms and patient statements),
- Objective (clinical findings and observations),
- Assessment (diagnosis or potential diagnoses),
- Plan (treatment and follow-up).
Ensure the note is concise, clear, and accurately reflects the conversation.
"""
conversation = [
{"role": "system", "content": system_prompt},
{"role": "user", "content": f"{task_prompt}\n\nTranscribed conversation:\n{transcribed_text}"}
]
input_ids = tokenizer.apply_chat_template(conversation, return_tensors="pt")
if input_ids.shape[1] > MAX_INPUT_TOKEN_LENGTH:
input_ids = input_ids[:, -MAX_INPUT_TOKEN_LENGTH:]
gr.Warning(f"Trimmed input from conversation as it was longer than {MAX_INPUT_TOKEN_LENGTH} tokens.")
input_ids = input_ids.to(llm.device)
streamer = TextIteratorStreamer(tokenizer, timeout=10.0, skip_prompt=True, skip_special_tokens=True)
generate_kwargs = dict(
{"input_ids": input_ids},
streamer=streamer,
max_new_tokens=max_new_tokens,
do_sample=True,
top_p=top_p,
top_k=top_k,
temperature=temperature,
num_beams=1,
repetition_penalty=repetition_penalty,
)
t = Thread(target=llm.generate, kwargs=generate_kwargs)
t.start()
outputs = []
for text in streamer:
outputs.append(text)
yield "".join(outputs)
# Gradio Interface combining transcription and SOAP note generation
demo = gr.Blocks(theme=gr.themes.Ocean())
with demo:
with gr.Tab("Clinical SOAP Note from Audio"):
# Transcription Interface
audio_input = gr.Audio(sources=["microphone", "upload"], type="filepath", label="Audio Input")
task_input = gr.Radio(["transcribe", "translate"], label="Task", value="transcribe")
transcription_output = gr.Textbox(label="Transcription Output")
# Transcription button
transcribe_button = gr.Button("Transcribe")
transcribe_button.click(fn=transcribe, inputs=[audio_input, task_input], outputs=transcription_output)
# SOAP Generation Interface
transcribed_text_input = gr.Textbox(label="Edit Transcription before SOAP Generation", lines=5)
system_prompt_input = gr.Textbox(label="System Prompt", lines=2, value="You are a world class clinical assistant.")
max_new_tokens_input = gr.Slider(label="Max new tokens", minimum=1, maximum=2048, value=1024, step=1)
temperature_input = gr.Slider(label="Temperature", minimum=0.1, maximum=4.0, value=0.6, step=0.1)
top_p_input = gr.Slider(label="Top-p", minimum=0.05, maximum=1.0, value=0.9, step=0.05)
top_k_input = gr.Slider(label="Top-k", minimum=1, maximum=1000, value=50, step=1)
repetition_penalty_input = gr.Slider(label="Repetition penalty", minimum=1.0, maximum=2.0, value=1.2, step=0.05)
soap_output = gr.Textbox(label="Generated SOAP Note Output")
# SOAP generation button
generate_soap_button = gr.Button("Generate SOAP Note")
generate_soap_button.click(
fn=generate_soap,
inputs=[
transcribed_text_input,
system_prompt_input,
max_new_tokens_input,
temperature_input,
top_p_input,
top_k_input,
repetition_penalty_input
],
outputs=soap_output
)
# Automatically copy transcription output to the edit box
def update_transcription_box(transcription_text):
return transcription_text
transcription_output.change(fn=update_transcription_box, inputs=transcription_output, outputs=transcribed_text_input)
demo.queue().launch(ssr_mode=False)