diff --git a/.gitignore b/.gitignore index da03d3f..c0e95ec 100644 --- a/.gitignore +++ b/.gitignore @@ -170,5 +170,6 @@ _.aifs software/output_audio.wav .DS_Store +# ignore node modules and .expo files node_modules/ -.expo/ \ No newline at end of file +.expo/ diff --git a/software/source/clients/ios/react-native/src/screens/Main.tsx b/software/source/clients/ios/react-native/src/screens/Main.tsx index d6a69cb..a9a5ed3 100644 --- a/software/source/clients/ios/react-native/src/screens/Main.tsx +++ b/software/source/clients/ios/react-native/src/screens/Main.tsx @@ -1,7 +1,7 @@ import React, { useState, useEffect } from "react"; import { View, Text, TouchableOpacity, StyleSheet } from "react-native"; import * as FileSystem from 'expo-file-system'; -import { Audio } from "expo-av"; +import { AVPlaybackStatus, AVPlaybackStatusSuccess, Audio } from "expo-av"; interface MainProps { route: { @@ -19,30 +19,14 @@ const Main: React.FC = ({ route }) => { const [audioQueue, setAudioQueue] = useState([]); const [sound, setSound] = useState(); const audioDir = FileSystem.documentDirectory + '01/audio/'; - const Buffer = require('buffer').Buffer; - const toBuffer = async (blob: Blob) => { + const constructTempFilePath = async (buffer: string) => { + await dirExists(); - const uri = await toDataURI(blob); - const base64 = uri.replace(/^.*,/g, ""); - return Buffer.from(base64, "base64"); - }; - - const toDataURI = (blob: Blob) => - new Promise((resolve) => { - const reader = new FileReader(); - reader.readAsDataURL(blob); - reader.onloadend = () => { - const uri = reader.result?.toString(); - resolve(uri); - }; - }); - - const constructTempFilePath = async (buffer: Buffer) => { const tempFilePath = `${audioDir}${Date.now()}.wav`; await FileSystem.writeAsStringAsync( tempFilePath, - buffer.toString(), + buffer, { encoding: FileSystem.EncodingType.Base64, } @@ -66,6 +50,12 @@ const Main: React.FC = ({ route }) => { const playNextAudio = async () => { console.log("in playNextAudio audioQueue is", audioQueue.length); + if (sound != null){ + console.log('Unloading Sound'); + await sound.unloadAsync(); + setSound(null); + } + if (audioQueue.length > 0) { const uri = audioQueue.shift() as string; console.log("load audio from", uri); @@ -76,6 +66,9 @@ const Main: React.FC = ({ route }) => { console.log("playing audio from", uri); await sound?.playAsync(); + + sound.setOnPlaybackStatusUpdate(_onPlayBackStatusUpdate); + } catch (error){ console.log("Error playing audio", error); playNextAudio(); @@ -84,16 +77,17 @@ const Main: React.FC = ({ route }) => { } }; - useEffect(() => { - return sound - ? () => { - console.log('Unloading Sound'); - sound.unloadAsync(); - setSound(null); - playNextAudio(); - } - : undefined; - }, [sound]); + const isAVPlaybackStatusSuccess = ( + status: AVPlaybackStatus + ): status is AVPlaybackStatusSuccess => { + return (status as AVPlaybackStatusSuccess).isLoaded !== undefined; + }; + + const _onPlayBackStatusUpdate = (status: AVPlaybackStatus) => { + if (isAVPlaybackStatusSuccess(status) && status.didJustFinish){ + playNextAudio(); + } + } useEffect(() => { console.log("audioQueue has been updated:", audioQueue.length); @@ -115,15 +109,16 @@ const Main: React.FC = ({ route }) => { }; websocket.onmessage = async (e) => { - console.log("Received message from WebSocket", e.data); - const blob = await e.data; - const buffer = await toBuffer(blob); + const message = JSON.parse(e.data); + console.log(message.content); + + const buffer = await message.content; const filePath = await constructTempFilePath(buffer); setAudioQueue((prevQueue) => [...prevQueue, filePath]); console.log("audio file written to", filePath); - if (e.data.format === "bytes.raw" && e.data.end && audioQueue.length > 1) { + if (message.format === "bytes.raw" && message.end && audioQueue.length >= 1) { console.log("calling playNextAudio"); playNextAudio(); } diff --git a/software/source/server/server.py b/software/source/server/server.py index dd71b06..a347026 100644 --- a/software/source/server/server.py +++ b/software/source/server/server.py @@ -393,8 +393,10 @@ def stream_tts(sentence): with open(audio_file, "rb") as f: audio_bytes = f.read() - desktop_path = os.path.join(os.path.expanduser('~'), 'Desktop') - desktop_audio_file = os.path.join(desktop_path, os.path.basename(audio_file)) + desktop_path = os.path.join(os.path.expanduser("~"), "Desktop") + desktop_audio_file = os.path.join( + desktop_path, f"{datetime.datetime.now()}" + os.path.basename(audio_file) + ) shutil.copy(audio_file, desktop_audio_file) print(f"Audio file saved to Desktop: {desktop_audio_file}") # storage_client = storage.Client(project="react-native-421323") @@ -409,15 +411,23 @@ def stream_tts(sentence): # f"Audio file {audio_file} uploaded to {datetime.datetime.now().strftime('%Y%m%d%H%M%S%f')}.wav" # ) - os.remove(audio_file) - file_type = "audio/wav" # Read the entire WAV file with open(audio_file, "rb") as f: audio_bytes = f.read() + os.remove(audio_file) + # Stream the audio as a single message - yield {"role": "assistant", "type": "audio", "format": file_type, "content": base64.b64encode(audio_bytes).decode('utf-8'), "start": True, "end": True} + yield { + "role": "assistant", + "type": "audio", + "format": file_type, + "content": base64.b64encode(audio_bytes).decode("utf-8"), + "start": True, + "end": True, + } + from uvicorn import Config, Server import os