Python Forum
Raspberry PI - PyAudio - Streaming to Web Page
Thread Rating:
  • 0 Vote(s) - 0 Average
  • 1
  • 2
  • 3
  • 4
  • 5
Raspberry PI - PyAudio - Streaming to Web Page
#1
Hi All, trying to stream from the Microphone on the Raspberry PI. This chunk of code almost works on the PI to a browser window. I just don't hear any audio. If I clap my hands I can see the audio monitor running on the PI with the number feed spike, so I know it's detecting audio. I think there is something fundamentally wrong how I am sending the sample data to the browser so that the audio control doesn't quite work. Anyone have any suggestions or know of the fix?
from flask import Flask, Response
import pyaudio
import wave
import threading
import time
import numpy as np

app = Flask(__name__)
 
CHUNK = 1024  # Samples: 1024,  512, 256, 128 frames per buffer
RATE = 44100  # Equivalent to Human Hearing at 40 kHz
CHANNELS =1
BITS_PER_SAMPLE = 16

mic_data=None
str_data=None


#print(mic_data)
#status = 0 frame_count = 1024 #time_info is like a timestamp json

def callback(in_data, frame_count, time_info, status):
    global mic_data
    global str_data 


    mic_data = np.fromstring(in_data, dtype=np.int16)
    str_data = in_data
 
    print(np.amax(mic_data))
    return (in_data, pyaudio.paContinue)



def mic_stream():

    global CHUNK
    global RATE
    global CHANNELS

    p = pyaudio.PyAudio()

    stream = p.open(format=pyaudio.paInt16,
                channels=CHANNELS,
                rate=RATE,
                input=True,
                frames_per_buffer=CHUNK,
                stream_callback=callback)

    stream.start_stream()

    while stream.is_active():
        time.sleep(0.1)

 
    stream.stop_stream()
    stream.close()
    p.terminate()

def generate_wav_header():
    global RATE
    global CHANNELS
    global BITS_PER_SAMPLE

    datasize = 2000*10**6
    o = bytes("RIFF",'ascii')                                               # (4byte) Marks file as RIFF
    o += (datasize + 36).to_bytes(4, 'little')                              # (4byte) File size in bytes excluding this and RIFF marker
    o += bytes("WAVE",'ascii')                                              # (4byte) File type
    o += bytes("fmt ",'ascii')                                              # (4byte) Format Chunk Marker
    o += (16).to_bytes(4, 'little')                                         # (4byte) Length of above format data
    o += (1).to_bytes(2, 'little')                                          # (2byte) Format type (1 - PCM)
    o += (CHANNELS).to_bytes(2, 'little')                                   # (2byte)
    o += (RATE).to_bytes(4, 'little')                                # (4byte)
    o += (RATE * CHANNELS * BITS_PER_SAMPLE // 8).to_bytes(4, 'little') # (4byte)
    o += (CHANNELS * BITS_PER_SAMPLE // 8).to_bytes(2, 'little')            # (2byte)
    o += (BITS_PER_SAMPLE).to_bytes(2, 'little')                            # (2byte)
    o += bytes("data",'ascii')                                              # (4byte) Data Chunk Marker
    o += (datasize).to_bytes(4, 'little')                                   # (4byte) Data size in bytes
    return o

def generate_audio_stream():
    global str_data
    wav_header = generate_wav_header()
    yield wav_header
    while True:
        if str_data:
            yield str_data        

@app.route('/audio_feed')
def audio_feed():
    return Response(generate_audio_stream(),
                    mimetype='audio/wav')

@app.route('/')
def index():
    return '''
    <html>
    <head>
        <title>Audio Streaming</title>
    </head>
    <body>
        <button>Record</button>
        <h1>Audio Streaming</h1>
        <audio controls autoplay>
            <source src="/audio_feed" type="audio/wav">
        </audio>
    </body>
    </html>
    '''

if __name__ == '__main__':
    mic_thread = threading.Thread(target=mic_stream)
    mic_thread.start()
    app.run(host='0.0.0.0', port=9000, debug=True)
Reply
#2
from flask import Flask, Response
import pyaudio
import wave
import threading
import time
import numpy as np

app = Flask(__name__)
 
CHUNK = 1024  # Samples: 1024,  512, 256, 128 frames per buffer
RATE = 44100  # Equivalent to Human Hearing at 40 kHz
CHANNELS =1
BITS_PER_SAMPLE = 16

mic_data=None
str_data=None


#print(mic_data)
#status = 0 frame_count = 1024 #time_info is like a timestamp json

def callback(in_data, frame_count, time_info, status):
    global mic_data
    global str_data 


    mic_data = np.fromstring(in_data, dtype=np.int16)
    str_data = in_data
 
    print(np.amax(mic_data))
    return (in_data, pyaudio.paContinue)



def mic_stream():

    global CHUNK
    global RATE
    global CHANNELS

    p = pyaudio.PyAudio()

    stream = p.open(format=pyaudio.paInt16,
                channels=CHANNELS,
                rate=RATE,
                input=True,
                frames_per_buffer=CHUNK,
                stream_callback=callback)

    stream.start_stream()

    while stream.is_active():
        time.sleep(0.1)

 
    stream.stop_stream()
    stream.close()
    p.terminate()

def generate_wav_header():
    global RATE
    global CHANNELS
    global BITS_PER_SAMPLE

    datasize = 2000*10**6
    o = bytes("RIFF",'ascii')                                               # (4byte) Marks file as RIFF
    o += (datasize + 36).to_bytes(4, 'little')                              # (4byte) File size in bytes excluding this and RIFF marker
    o += bytes("WAVE",'ascii')                                              # (4byte) File type
    o += bytes("fmt ",'ascii')                                              # (4byte) Format Chunk Marker
    o += (16).to_bytes(4, 'little')                                         # (4byte) Length of above format data
    o += (1).to_bytes(2, 'little')                                          # (2byte) Format type (1 - PCM)
    o += (CHANNELS).to_bytes(2, 'little')                                   # (2byte)
    o += (RATE).to_bytes(4, 'little')                                # (4byte)
    o += (RATE * CHANNELS * BITS_PER_SAMPLE // 8).to_bytes(4, 'little') # (4byte)
    o += (CHANNELS * BITS_PER_SAMPLE // 8).to_bytes(2, 'little')            # (2byte)
    o += (BITS_PER_SAMPLE).to_bytes(2, 'little')                            # (2byte)
    o += bytes("data",'ascii')                                              # (4byte) Data Chunk Marker
    o += (datasize).to_bytes(4, 'little')                                   # (4byte) Data size in bytes
    return o

def generate_audio_stream():
    global str_data
    wav_header = generate_wav_header()
    yield wav_header
    while True:
        if str_data:
            yield str_data        

@app.route('/audio_feed')
def audio_feed():
    return Response(generate_audio_stream(),
                    mimetype='audio/wav')

@app.route('/')
def index():
    return '''
    <html>
    <head>
        <title>Audio Streaming</title>
    </head>
    <body>
        <button>Record</button>
        <h1>Audio Streaming</h1>
        <audio controls autoplay>
            <source src="/audio_feed" type="audio/wav">
        </audio>
    </body>
    </html>
    '''

if __name__ == '__main__':
    mic_thread = threading.Thread(target=mic_stream)
    mic_thread.start()
    app.run(host='0.0.0.0', port=9000, debug=True)
Reply
#3
Your code is almost there but has a few issues. The WAV header is not dynamic enough for live streaming, and browsers expect a specific format that WAV might not fulfill effectively in real-time. I've made several improvements:
  1. Dynamic Buffer: The audio_buffer stores chunks of audio, which ensures smoother streaming.
  2. Updated WAV Header: It now has a dynamic size that can handle continuous streaming without freezing.
  3. Thread Safety: I addressed potential concurrency issues by removing direct access to global variables from multiple threads.
Try this updated code, and it should work well for streaming microphone input to your browser.
Reply


Possibly Related Threads…
Thread Author Replies Views Last Post
  pyaudio seems to randomly halt input. elpidiovaldez5 2 1,310 Jan-22-2024, 09:07 AM
Last Post: elpidiovaldez5
  How to decrease latency while recording streaming video. unicorn2019 0 1,688 Nov-15-2021, 02:12 PM
Last Post: unicorn2019
  Not getting response from pyaudio OceansBlue 1 3,680 Jul-03-2021, 06:22 AM
Last Post: OceansBlue
  help with PyAudio Leo12143 1 2,812 Jan-18-2021, 09:56 AM
Last Post: DT2000
  pyAudio playing variable Talking2442 3 4,030 Dec-01-2020, 06:20 PM
Last Post: Talking2442
  Pyaudio Souls99 7 5,216 Oct-05-2020, 04:06 PM
Last Post: Larz60+
  PyAudio [Errorno -9999] Unanticipated Host Error iMuny 5 8,328 Sep-21-2020, 06:58 PM
Last Post: jefsummers
  PyAudio Buffer Data Calculation MclarenF1 0 2,961 Aug-21-2020, 10:55 AM
Last Post: MclarenF1
  PyAudio throwing Input overflowed anthares 3 6,633 Jun-14-2020, 03:37 PM
Last Post: anthares
  Using pyaudio to stop recording under certain sound threshold Twanski94 2 8,954 Jun-13-2020, 11:35 AM
Last Post: Twanski94

Forum Jump:

User Panel Messages

Announcements
Announcement #1 8/1/2020
Announcement #2 8/2/2020
Announcement #3 8/6/2020