from fastapi.responses import PlainTextResponse from RealtimeSTT import AudioToTextRecorder from RealtimeTTS import TextToAudioStream import importlib import warnings import asyncio import types import wave import os import sys os.environ["INTERPRETER_REQUIRE_ACKNOWLEDGE"] = "False" def start_server(server_host, server_port, profile, voice, debug): # Load the profile module from the provided path spec = importlib.util.spec_from_file_location("profile", profile) profile_module = importlib.util.module_from_spec(spec) spec.loader.exec_module(profile_module) # Get the interpreter from the profile interpreter = profile_module.interpreter # Apply our settings to it interpreter.verbose = debug interpreter.server.host = server_host interpreter.server.port = server_port if voice == False: # If voice is False, just start the standard OI server interpreter.server.run() exit() # ONLY if voice is True, will we run the rest of this file. # STT interpreter.stt = AudioToTextRecorder( model="tiny.en", spinner=False, use_microphone=False ) interpreter.stt.stop() # It needs this for some reason # TTS if not hasattr(interpreter, 'tts'): print("Setting TTS provider to default: openai") interpreter.tts = "openai" if interpreter.tts == "coqui": from RealtimeTTS import CoquiEngine engine = CoquiEngine() elif interpreter.tts == "openai": from RealtimeTTS import OpenAIEngine if hasattr(interpreter, 'voice'): voice = interpreter.voice else: voice = "onyx" engine = OpenAIEngine(voice=voice) elif interpreter.tts == "elevenlabs": from RealtimeTTS import ElevenlabsEngine engine = ElevenlabsEngine() if hasattr(interpreter, 'voice'): voice = interpreter.voice else: voice = "Will" engine.set_voice(voice) else: raise ValueError(f"Unsupported TTS engine: {interpreter.tts}") interpreter.tts = TextToAudioStream(engine) # Misc Settings interpreter.play_audio = False interpreter.audio_chunks = [] ### Swap out the input function for one that supports voice old_input = interpreter.input async def new_input(self, chunk): await asyncio.sleep(0) if isinstance(chunk, bytes): self.stt.feed_audio(chunk) self.audio_chunks.append(chunk) elif isinstance(chunk, dict): if "start" in chunk: self.stt.start() self.audio_chunks = [] await old_input({"role": "user", "type": "message", "start": True}) if "end" in chunk: self.stt.stop() content = self.stt.text() if content.strip() == "": return print(">", content.strip()) if False: audio_bytes = bytearray(b"".join(self.audio_chunks)) with wave.open('audio.wav', 'wb') as wav_file: wav_file.setnchannels(1) wav_file.setsampwidth(2) # Assuming 16-bit audio wav_file.setframerate(16000) # Assuming 16kHz sample rate wav_file.writeframes(audio_bytes) print(os.path.abspath('audio.wav')) await old_input({"role": "user", "type": "message", "content": content}) await old_input({"role": "user", "type": "message", "end": True}) ### Swap out the output function for one that supports voice old_output = interpreter.output async def new_output(self): while True: output = await old_output() # if output == {"role": "assistant", "type": "message", "start": True}: # return {"role": "assistant", "type": "audio", "format": "bytes.wav", "start": True} if isinstance(output, bytes): return output await asyncio.sleep(0) delimiters = ".?!;,\n…)]}" if output["type"] == "message" and len(output.get("content", "")) > 0: self.tts.feed(output.get("content")) if not self.tts.is_playing() and any([c in delimiters for c in output.get("content")]): # Start playing once the first delimiter is encountered. self.tts.play_async(on_audio_chunk=self.on_tts_chunk, muted=not self.play_audio, sentence_fragment_delimiters=delimiters, minimum_sentence_length=9) return {"role": "assistant", "type": "audio", "format": "bytes.wav", "start": True} if output == {"role": "assistant", "type": "message", "end": True}: if not self.tts.is_playing(): # We put this here in case it never outputs a delimiter and never triggers play_async^ self.tts.play_async(on_audio_chunk=self.on_tts_chunk, muted=not self.play_audio, sentence_fragment_delimiters=delimiters, minimum_sentence_length=9) return {"role": "assistant", "type": "audio", "format": "bytes.wav", "start": True} return {"role": "assistant", "type": "audio", "format": "bytes.wav", "end": True} def on_tts_chunk(self, chunk): self.output_queue.sync_q.put(chunk) # Set methods on interpreter object interpreter.input = types.MethodType(new_input, interpreter) interpreter.output = types.MethodType(new_output, interpreter) interpreter.on_tts_chunk = types.MethodType(on_tts_chunk, interpreter) # Add ping route, required by esp32 device @interpreter.server.app.get("/ping") async def ping(): return PlainTextResponse("pong") # Start server interpreter.server.display = True interpreter.print = True interpreter.server.run()