NAME
Audio::StreamGenerator - create a 'radio' stream by mixing ('cross fading') multiple audio sources (files or anything that can be converted to PCM audio) and sending it to a streaming server (like Icecast)
SYNOPSIS
use strict;
use warnings;
use Audio::StreamGenerator;
my $out_command = q~
ffmpeg -re -f s16le -acodec pcm_s16le -ac 2 -ar 44100 -i - \
-acodec libopus -ac 2 -b:a 160k -content_type application/ogg -format ogg icecast://source:hackme@localhost:8000/our_radio.opus \
-acodec libmp3lame -ac 2 -b:a 192k -content_type audio/mpeg icecast://source:hackme@localhost:8000/our_radio.mp3 \
-acodec aac -b:a 192k -ac 2 -content_type audio/aac icecast://source:hackme@localhost:8000/our_radio.aac
~;
my $out_fh;
open ($out_fh, '|-', $out_command);
sub get_new_source {
my $fullpath = '/path/to/some/audiofile.flac';
my @ffmpeg_cmd = (
'/usr/bin/ffmpeg',
'-i',
$fullpath,
'-loglevel', 'quiet',
'-f', 's16le',
'-acodec', 'pcm_s16le',
'-ac', '2',
'-ar', '44100',
'-'
);
open(my $source, '-|', @ffmpeg_cmd);
return $source;
}
sub run_every_second {
my $streamert = shift;
my $position = $streamert->get_elapsed_seconds();
print STDERR "now at position $position\r";
if ([-some external event happened-]) { # skip to the next song requested
$streamert->skip()
}
}
my $streamer = Audio::StreamGenerator->new(
out_fh => $out_fh,
get_new_source => \&get_new_source,
run_every_second => \&run_every_second,
);
$streamer->stream();
DESCRIPTION
This module creates a 'live' audio stream that can be broadcast using streaming technologies like Icecast or HTTP Live Streaming.
It creates one ongoing audio stream by mixing or 'crossfading' multiple sources (normally audio files).
Although there is nothing stopping you from using this to generate a file that can be played back later, its intended use is to create a 'radio' stream that can be streamed or 'broadcast' live on the internet.
The module takes raw PCM audio from a file handle as input, and outputs raw PCM audio to another file handle. This means that an external program is necessary to decode (mp3/flac/etc) source files, and to encode & stream the actual output. For both purposes, ffmpeg is recommended - but anything that can produce and/or receive raw PCM audio should do.
CONSTRUCTOR METHOD
my $streamer = Audio::StreamGenerator->new( %options );
Creates a new StreamGenerator object and returns it.
OPTIONS
The following options can be specified:
KEY DEFAULT MANDATORY
----------- ------- ---------
out_fh - yes
get_new_source - yes
run_every_second - no
normal_fade_seconds 5 no
skip_fade_seconds 3 no
sample_rate 44100 no
channels_amount 2 no
max_vol_before_mix_fraction 0.75 no
out_fh
The outgoing file handle - this is where the generated signed 16-bit little-endian PCM audio stream is sent to.
Note that StreamGenerator has no notion of time - if you don't slow it down, it will process data as fast as it can - which is faster than your listeners are able to play the stream. On Icecast, this will cause listeners to be disconnected because they are "too far behind".
This can be addressed by making sure that the out_fh process consumes the audio no faster than realtime.
If you are using ffmpeg, you can achieve this with its '-re' option.
Another possibility is to first pipe the data to a command like 'pv' to rate limit the data. An additional advantage of 'pv' is that it can also add a buffer between the StreamGenerator and the encoder, which can absorb any short delays that may occur when StreamGenerator is switching to a new track.
Example:
pv -q -L 176400 -B 3528000 | ffmpeg ...
This will tell pv to be quiet (no output to STDERR), to allow a maximum throughput of 44100 samples per second * 2 bytes per sample * 2 channels = 176400 bytes per second, and keep a buffer of 176400 Bps * 20 seconds = 3528000 bytes
The out_fh command is also the place where you could insert a sound processing solution like the command line version of Stereo tool - just pipe the audio first to that tool, and from there to your encoder.
get_new_source
Reference to a sub that will be called every time that a new source (audio file) is needed. Needs to return a readable filehandle that will output signed 16-bit little-endian PCM audio.
run_every_second
This sub will be run after each second of playback, with the StreamGenerator object as an argument. This can be used to do things like updating a UI with the current playing position - or to call the skip() method if we need to skip to the next source.
normal_fade_seconds
Amount of seconds that we want tracks to overlap. This is only the initial/max value - the mixing algorithm may decide to mix less seconds if the old track ends with loud samples.
skip_fade_seconds
When 'skipping' to the next song using the skip() method (for example, after a user clicked a "next song" button on some web interface), we mix less seconds than normally, simply because mixing 5+ seconds in the middle of the old track sounds pretty bad. This value has to be lower than normal_fade_seconds.
sample_rate
The amount of samples per second (both incoming & outgoing), normally this is 44100 for standard CD-quality audio.
channels_amount
Amount of audio channels, this is normally 2 (stereo).
max_vol_before_mix_fraction
This tells StreamGenerator what the minimum volume of a 'loud' sample is. It is expressed as a fraction of the maximum volume. When mixing 2 tracks, StreamGenerator needs to find out what the last loud sample of the old track is so that it can start the next song immediately after that.
METHODS
stream
$streamer->stream();
Start the actual audio stream.
skip $streamer->skip();
Skip to the next track without finishing the current one. This can be called from the "run_every_second" sub, for example after checking whether a 'skip' flag was set in a database, or whether a file exists.
get_elapsed_samples
my $elapsed_samples = $streamer->get_elapsed_samples();
print "$elapsed_samples played so far\r";
Get the amount of played samples in the current track - this can be called from the "run_every_second" sub.
get_elapsed_seconds
my $elapsed_seconds = $streamer->get_elapsed_seconds();
print "now at position $elapsed_seconds of the current track\r";
Get the amount of elapsed seconds in the current track - in other words the current position in the track. This equals to get_elapsed_samples/sample_rate .