|
|
@ -1,6 +1,7 @@
|
|
|
|
from dotenv import load_dotenv
|
|
|
|
from dotenv import load_dotenv
|
|
|
|
load_dotenv() # take environment variables from .env.
|
|
|
|
load_dotenv() # take environment variables from .env.
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
import traceback
|
|
|
|
from platformdirs import user_data_dir
|
|
|
|
from platformdirs import user_data_dir
|
|
|
|
import ast
|
|
|
|
import ast
|
|
|
|
import json
|
|
|
|
import json
|
|
|
@ -183,136 +184,138 @@ async def send_messages(websocket: WebSocket):
|
|
|
|
async def listener():
|
|
|
|
async def listener():
|
|
|
|
|
|
|
|
|
|
|
|
while True:
|
|
|
|
while True:
|
|
|
|
while True:
|
|
|
|
try:
|
|
|
|
if not from_user.empty():
|
|
|
|
while True:
|
|
|
|
chunk = await from_user.get()
|
|
|
|
if not from_user.empty():
|
|
|
|
break
|
|
|
|
chunk = await from_user.get()
|
|
|
|
elif not from_computer.empty():
|
|
|
|
break
|
|
|
|
chunk = from_computer.get()
|
|
|
|
elif not from_computer.empty():
|
|
|
|
break
|
|
|
|
chunk = from_computer.get()
|
|
|
|
await asyncio.sleep(1)
|
|
|
|
break
|
|
|
|
|
|
|
|
await asyncio.sleep(1)
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
message = accumulator.accumulate(chunk)
|
|
|
|
|
|
|
|
if message == None:
|
|
|
|
|
|
|
|
# Will be None until we have a full message ready
|
|
|
|
|
|
|
|
continue
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
# print(str(message)[:1000])
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
# At this point, we have our message
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if message["type"] == "audio" and message["format"].startswith("bytes"):
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if not message["content"]: # If it was nothing / silence
|
|
|
|
message = accumulator.accumulate(chunk)
|
|
|
|
|
|
|
|
if message == None:
|
|
|
|
|
|
|
|
# Will be None until we have a full message ready
|
|
|
|
continue
|
|
|
|
continue
|
|
|
|
|
|
|
|
|
|
|
|
# Convert bytes to audio file
|
|
|
|
# print(str(message)[:1000])
|
|
|
|
# Format will be bytes.wav or bytes.opus
|
|
|
|
|
|
|
|
mime_type = "audio/" + message["format"].split(".")[1]
|
|
|
|
|
|
|
|
audio_file_path = bytes_to_wav(message["content"], mime_type)
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
# For microphone debugging:
|
|
|
|
|
|
|
|
if False:
|
|
|
|
|
|
|
|
os.system(f"open {audio_file_path}")
|
|
|
|
|
|
|
|
import time
|
|
|
|
|
|
|
|
time.sleep(15)
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
text = stt(audio_file_path)
|
|
|
|
|
|
|
|
print("> ", text)
|
|
|
|
|
|
|
|
message = {"role": "user", "type": "message", "content": text}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
# At this point, we have only text messages
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if type(message["content"]) != str:
|
|
|
|
# At this point, we have our message
|
|
|
|
print("This should be a string, but it's not:", message["content"])
|
|
|
|
|
|
|
|
message["content"] = message["content"].decode()
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
# Custom stop message will halt us
|
|
|
|
if message["type"] == "audio" and message["format"].startswith("bytes"):
|
|
|
|
if message["content"].lower().strip(".,! ") == "stop":
|
|
|
|
|
|
|
|
continue
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
# Load, append, and save conversation history
|
|
|
|
if "content" not in message or message["content"] == None: # If it was nothing / silence
|
|
|
|
with open(conversation_history_path, 'r') as file:
|
|
|
|
continue
|
|
|
|
messages = json.load(file)
|
|
|
|
|
|
|
|
messages.append(message)
|
|
|
|
|
|
|
|
with open(conversation_history_path, 'w') as file:
|
|
|
|
|
|
|
|
json.dump(messages, file, indent=4)
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
accumulated_text = ""
|
|
|
|
# Convert bytes to audio file
|
|
|
|
|
|
|
|
# Format will be bytes.wav or bytes.opus
|
|
|
|
|
|
|
|
mime_type = "audio/" + message["format"].split(".")[1]
|
|
|
|
|
|
|
|
audio_file_path = bytes_to_wav(message["content"], mime_type)
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
# For microphone debugging:
|
|
|
|
|
|
|
|
if False:
|
|
|
|
|
|
|
|
os.system(f"open {audio_file_path}")
|
|
|
|
|
|
|
|
import time
|
|
|
|
|
|
|
|
time.sleep(15)
|
|
|
|
|
|
|
|
|
|
|
|
if any([m["type"] == "image" for m in messages]) and interpreter.llm.model.startswith("gpt-"):
|
|
|
|
text = stt(audio_file_path)
|
|
|
|
interpreter.llm.model = "gpt-4-vision-preview"
|
|
|
|
print("> ", text)
|
|
|
|
interpreter.llm.supports_vision = True
|
|
|
|
message = {"role": "user", "type": "message", "content": text}
|
|
|
|
|
|
|
|
|
|
|
|
for chunk in interpreter.chat(messages, stream=True, display=True):
|
|
|
|
# At this point, we have only text messages
|
|
|
|
|
|
|
|
|
|
|
|
if any([m["type"] == "image" for m in interpreter.messages]):
|
|
|
|
if type(message["content"]) != str:
|
|
|
|
interpreter.llm.model = "gpt-4-vision-preview"
|
|
|
|
print("This should be a string, but it's not:", message["content"])
|
|
|
|
|
|
|
|
message["content"] = message["content"].decode()
|
|
|
|
|
|
|
|
|
|
|
|
logger.debug("Got chunk:", chunk)
|
|
|
|
# Custom stop message will halt us
|
|
|
|
|
|
|
|
if message["content"].lower().strip(".,! ") == "stop":
|
|
|
|
|
|
|
|
continue
|
|
|
|
|
|
|
|
|
|
|
|
# Send it to the user
|
|
|
|
# Load, append, and save conversation history
|
|
|
|
await to_device.put(chunk)
|
|
|
|
with open(conversation_history_path, 'r') as file:
|
|
|
|
# Yield to the event loop, so you actually send it out
|
|
|
|
messages = json.load(file)
|
|
|
|
await asyncio.sleep(0.01)
|
|
|
|
messages.append(message)
|
|
|
|
|
|
|
|
with open(conversation_history_path, 'w') as file:
|
|
|
|
|
|
|
|
json.dump(messages, file, indent=4)
|
|
|
|
|
|
|
|
|
|
|
|
if os.getenv('TTS_RUNNER') == "server":
|
|
|
|
accumulated_text = ""
|
|
|
|
# Speak full sentences out loud
|
|
|
|
|
|
|
|
if chunk["role"] == "assistant" and "content" in chunk and chunk["type"] == "message":
|
|
|
|
|
|
|
|
accumulated_text += chunk["content"]
|
|
|
|
|
|
|
|
sentences = split_into_sentences(accumulated_text)
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
# If we're going to speak, say we're going to stop sending text.
|
|
|
|
|
|
|
|
# This should be fixed probably, we should be able to do both in parallel, or only one.
|
|
|
|
|
|
|
|
if any(is_full_sentence(sentence) for sentence in sentences):
|
|
|
|
|
|
|
|
await to_device.put({"role": "assistant", "type": "message", "end": True})
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if is_full_sentence(sentences[-1]):
|
|
|
|
if any([m["type"] == "image" for m in messages]) and interpreter.llm.model.startswith("gpt-"):
|
|
|
|
for sentence in sentences:
|
|
|
|
interpreter.llm.model = "gpt-4-vision-preview"
|
|
|
|
await stream_tts_to_device(sentence)
|
|
|
|
interpreter.llm.supports_vision = True
|
|
|
|
accumulated_text = ""
|
|
|
|
|
|
|
|
|
|
|
|
for chunk in interpreter.chat(messages, stream=True, display=True):
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if any([m["type"] == "image" for m in interpreter.messages]):
|
|
|
|
|
|
|
|
interpreter.llm.model = "gpt-4-vision-preview"
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
logger.debug("Got chunk:", chunk)
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
# Send it to the user
|
|
|
|
|
|
|
|
await to_device.put(chunk)
|
|
|
|
|
|
|
|
# Yield to the event loop, so you actually send it out
|
|
|
|
|
|
|
|
await asyncio.sleep(0.01)
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if os.getenv('TTS_RUNNER') == "server":
|
|
|
|
|
|
|
|
# Speak full sentences out loud
|
|
|
|
|
|
|
|
if chunk["role"] == "assistant" and "content" in chunk and chunk["type"] == "message":
|
|
|
|
|
|
|
|
accumulated_text += chunk["content"]
|
|
|
|
|
|
|
|
sentences = split_into_sentences(accumulated_text)
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
# If we're going to speak, say we're going to stop sending text.
|
|
|
|
|
|
|
|
# This should be fixed probably, we should be able to do both in parallel, or only one.
|
|
|
|
|
|
|
|
if any(is_full_sentence(sentence) for sentence in sentences):
|
|
|
|
|
|
|
|
await to_device.put({"role": "assistant", "type": "message", "end": True})
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if is_full_sentence(sentences[-1]):
|
|
|
|
|
|
|
|
for sentence in sentences:
|
|
|
|
|
|
|
|
await stream_tts_to_device(sentence)
|
|
|
|
|
|
|
|
accumulated_text = ""
|
|
|
|
|
|
|
|
else:
|
|
|
|
|
|
|
|
for sentence in sentences[:-1]:
|
|
|
|
|
|
|
|
await stream_tts_to_device(sentence)
|
|
|
|
|
|
|
|
accumulated_text = sentences[-1]
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
# If we're going to speak, say we're going to stop sending text.
|
|
|
|
|
|
|
|
# This should be fixed probably, we should be able to do both in parallel, or only one.
|
|
|
|
|
|
|
|
if any(is_full_sentence(sentence) for sentence in sentences):
|
|
|
|
|
|
|
|
await to_device.put({"role": "assistant", "type": "message", "start": True})
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
# If we have a new message, save our progress and go back to the top
|
|
|
|
|
|
|
|
if not from_user.empty():
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
# Check if it's just an end flag. We ignore those.
|
|
|
|
|
|
|
|
temp_message = await from_user.get()
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if type(temp_message) is dict and temp_message.get("role") == "user" and temp_message.get("end"):
|
|
|
|
|
|
|
|
# Yup. False alarm.
|
|
|
|
|
|
|
|
continue
|
|
|
|
else:
|
|
|
|
else:
|
|
|
|
for sentence in sentences[:-1]:
|
|
|
|
# Whoops! Put that back
|
|
|
|
await stream_tts_to_device(sentence)
|
|
|
|
await from_user.put(temp_message)
|
|
|
|
accumulated_text = sentences[-1]
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
# If we're going to speak, say we're going to stop sending text.
|
|
|
|
|
|
|
|
# This should be fixed probably, we should be able to do both in parallel, or only one.
|
|
|
|
|
|
|
|
if any(is_full_sentence(sentence) for sentence in sentences):
|
|
|
|
|
|
|
|
await to_device.put({"role": "assistant", "type": "message", "start": True})
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
# If we have a new message, save our progress and go back to the top
|
|
|
|
|
|
|
|
if not from_user.empty():
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
# Check if it's just an end flag. We ignore those.
|
|
|
|
with open(conversation_history_path, 'w') as file:
|
|
|
|
temp_message = await from_user.get()
|
|
|
|
json.dump(interpreter.messages, file, indent=4)
|
|
|
|
|
|
|
|
|
|
|
|
if type(temp_message) is dict and temp_message.get("role") == "user" and temp_message.get("end"):
|
|
|
|
# TODO: is triggering seemingly randomly
|
|
|
|
# Yup. False alarm.
|
|
|
|
#logger.info("New user message recieved. Breaking.")
|
|
|
|
continue
|
|
|
|
#break
|
|
|
|
else:
|
|
|
|
|
|
|
|
# Whoops! Put that back
|
|
|
|
|
|
|
|
await from_user.put(temp_message)
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
with open(conversation_history_path, 'w') as file:
|
|
|
|
|
|
|
|
json.dump(interpreter.messages, file, indent=4)
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
# TODO: is triggering seemingly randomly
|
|
|
|
|
|
|
|
#logger.info("New user message recieved. Breaking.")
|
|
|
|
|
|
|
|
#break
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
# Also check if there's any new computer messages
|
|
|
|
# Also check if there's any new computer messages
|
|
|
|
if not from_computer.empty():
|
|
|
|
if not from_computer.empty():
|
|
|
|
|
|
|
|
|
|
|
|
with open(conversation_history_path, 'w') as file:
|
|
|
|
with open(conversation_history_path, 'w') as file:
|
|
|
|
json.dump(interpreter.messages, file, indent=4)
|
|
|
|
json.dump(interpreter.messages, file, indent=4)
|
|
|
|
|
|
|
|
|
|
|
|
logger.info("New computer message recieved. Breaking.")
|
|
|
|
|
|
|
|
break
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
logger.info("New computer message recieved. Breaking.")
|
|
|
|
|
|
|
|
break
|
|
|
|
|
|
|
|
except:
|
|
|
|
|
|
|
|
traceback.print_exc()
|
|
|
|
|
|
|
|
|
|
|
|
async def stream_tts_to_device(sentence):
|
|
|
|
async def stream_tts_to_device(sentence):
|
|
|
|
force_task_completion_responses = [
|
|
|
|
force_task_completion_responses = [
|
|
|
|