#!/usr/bin/perl -w ########################################################################### =head1 NAME Shout - Perl glue for libshout MP3 streaming source library =head1 SYNOPSIS use Shout qw{}; my $conn = new Shout host => 'localhost', port => 8000, mount => 'testing', nonblocking => 0, password => 'pa$$word!', user => 'username', dumpfile => undef, name => 'Wir sind nur Affen', url => 'http://apan.org/' genre => 'Monkey Music', description => 'A whole lotta monkey music.', format => SHOUT_FORMAT_MP3, protocol => SHOUT_PROTOCOL_HTTP, public => 0; # - or - my $conn = new Shout; $conn->host('localhost'); $conn->port(8000); $conn->mount('testing'); $conn->nonblocking(0); $conn->password('pa$$word!'); $conn->user('username'); $conn->dumpfile(undef); $conn->name('Test libshout-perl stream'); $conn->url('http://www.icecast.org/'); $conn->genre('perl'); $conn->format(SHOUT_FORMAT_MP3); $conn->protocol(SHOUT_PROTOCOL_HTTP); $conn->description('Stream with icecast at http://www.icecast.org'); $conn->public(0); ### Set your stream audio parameters for YP if you want $conn->set_audio_info(SHOUT_AI_BITRATE => 128, SHOUT_AI_SAMPLERATE => 44100); ### Connect to the server $conn->open or die "Failed to open: ", $conn->get_error; ### Set stream info $conn->set_metadata('song' => 'Scott Joplin - Maple Leaf Rag'); ### Stream some data my ( $buffer, $bytes ) = ( '', 0 ); while( ($bytes = sysread( STDIN, $buffer, 4096 )) > 0 ) { $conn->send( $buffer ) && next; print STDERR "Error while sending: ", $conn->get_error, "\n"; last; } continue { $conn->sync } ### Now close the connection $conn->close; =head1 EXPORTS Nothing by default. =head2 :constants The following error constants are exported into your package if the 'C<:constants>' tag is given as an argument to the C statement. SHOUT_FORMAT_MP3 SHOUT_FORMAT_VORBIS SHOUT_PROTOCOL_ICY SHOUT_PROTOCOL_XAUDIOCAST SHOUT_PROTOCOL_HTTP SHOUT_AI_BITRATE SHOUT_AI_SAMPLERATE SHOUT_AI_CHANNELS SHOUT_AI_QUALITY =head2 :functions The following functions are exported into your package if the 'C<:functions>' tag is given as an argument to the C statement. shout_open shout_close shout_metadata_new shout_metadata_free shout_metadata_add shout_set_metadata shout_send_data shout_sync shout_delay shout_set_host shout_set_port shout_set_mount shout_set_nonblocking shout_set_password shout_set_user shout_set_icq shout_set_irc shout_set_dumpfile shout_set_name shout_set_url shout_set_genre shout_set_description shout_set_public shout_get_host shout_get_port shout_get_mount shout_get_nonblocking shout_get_password shout_get_user shout_get_icq shout_get_irc shout_get_dumpfile shout_get_name shout_get_url shout_get_genre shout_get_description shout_get_public shout_get_error shout_get_errno shout_set_format shout_get_format shout_set_protocol shout_get_protocol shout_set_audio_info shout_get_audio_info shout_queuelen They work almost identically to their libshout C counterparts. See the libshout documentation for more information about how to use the function interface. =head2 :all All of the above symbols can be imported into your namespace by giving the 'C<:all>' tag as an argument to the C statement. =head1 DESCRIPTION This module is an object-oriented interface to libshout, an Ogg Vorbis and MP3 streaming library that allows applications to easily communicate and broadcast to an Icecast streaming media server. It handles the socket connections, metadata communication, and data streaming for the calling application, and lets developers focus on feature sets instead of implementation details. =head1 AUTHOR Jack Moffitt Paul Bournival (update to icecast 2) Brendan Cully (current maintainer) =cut ########################################################################### package Shout; use strict; BEGIN { use vars qw($VERSION @ISA @EXPORT @EXPORT_OK %EXPORT_TAGS $AUTOLOAD); $VERSION = '2.1'; use Carp; require Exporter; require DynaLoader; require AutoLoader; # Inheritance @ISA = qw(Exporter DynaLoader); ### Exporter stuff @EXPORT = qw { SHOUT_FORMAT_VORBIS SHOUT_FORMAT_MP3 SHOUT_PROTOCOL_ICY SHOUT_PROTOCOL_XAUDIOCAST SHOUT_PROTOCOL_HTTP SHOUT_AI_BITRATE SHOUT_AI_SAMPLERATE SHOUT_AI_QUALITY SHOUT_AI_CHANNELS SHOUTERR_SUCCESS SHOUTERR_INSANE SHOUTERR_NOCONNECT SHOUTERR_NOLOGIN SHOUTERR_SOCKET SHOUTERR_MALLOC SHOUTERR_METADATA SHOUTERR_CONNECTED SHOUTERR_UNCONNECTED SHOUTERR_UNSUPPORTED SHOUTERR_BUSY }; @EXPORT_OK = qw{ SHOUT_FORMAT_VORBIS SHOUT_FORMAT_MP3 SHOUT_PROTOCOL_ICY SHOUT_PROTOCOL_XAUDIOCAST SHOUT_PROTOCOL_HTTP SHOUT_AI_BITRATE SHOUT_AI_SAMPLERATE SHOUT_AI_QUALITY SHOUT_AI_CHANNELS SHOUTERR_SUCCESS SHOUTERR_INSANE SHOUTERR_NOCONNECT SHOUTERR_NOLOGIN SHOUTERR_SOCKET SHOUTERR_MALLOC SHOUTERR_METADATA SHOUTERR_CONNECTED SHOUTERR_UNCONNECTED SHOUTERR_UNSUPPORTED SHOUTERR_BUSY shout_open shout_close shout_set_metadata shout_metadata_new shout_metadata_free shout_metadata_add shout_send_data shout_sync shout_delay shout_queuelen shout_set_host shout_set_port shout_set_mount shout_set_password shout_set_user shout_set_dumpfile shout_set_name shout_set_url shout_set_genre shout_set_description shout_set_public shout_get_host shout_get_port shout_get_mount shout_get_password shout_get_user shout_get_dumpfile shout_get_name shout_get_url shout_get_genre shout_get_description shout_get_public shout_get_error shout_set_format shout_get_format shout_get_audio_info shout_set_audio_info shout_set_protocol shout_get_protocol shout_get_errno shout_get_nonblocking shout_set_nonblocking }; %EXPORT_TAGS = ( all => \@EXPORT_OK, constants => [qw{SHOUT_FORMAT_VORBIS SHOUT_FORMAT_MP3 SHOUT_PROTOCOL_ICY SHOUT_PROTOCOL_XAUDIOCAST SHOUT_PROTOCOL_HTTP SHOUT_AI_BITRATE SHOUT_AI_SAMPLERATE SHOUT_AI_QUALITY SHOUT_AI_CHANNELS SHOUTERR_SUCCESS SHOUTERR_INSANE SHOUTERR_NOCONNECT SHOUTERR_NOLOGIN SHOUTERR_SOCKET SHOUTERR_MALLOC SHOUTERR_METADATA SHOUTERR_CONNECTED SHOUTERR_UNCONNECTED SHOUTERR_UNSUPPORTED SHOUTERR_BUSY }], functions => [qw{shout_open shout_close shout_set_metadata shout_metadata_add shout_metadata_new shout_metadata_free shout_send_data shout_sync shout_delay shout_get_audio_info shout_set_audio_info shout_set_host shout_set_port shout_set_mount shout_set_password shout_set_user shout_set_dumpfile shout_set_name shout_set_url shout_set_genre shout_set_description shout_set_public shout_get_host shout_get_port shout_get_mount shout_get_password shout_get_user shout_get_dumpfile shout_get_name shout_get_url shout_get_genre shout_get_description shout_get_public shout_get_error shout_get_errno shout_set_protocol shout_get_protocol shout_set_format shout_get_format shout_get_nonblocking shout_set_nonblocking shout_queuelen }], ); } INIT { shout_init (); } END { shout_shutdown (); } bootstrap Shout $VERSION; ############################################################################### ### C O N F I G U R A T I O N G L O B A L S ############################################################################### use vars qw{@TranslatedMethods %CompatibilityMethods}; @TranslatedMethods = qw{ host port mount password user dumpfile name url genre description public format protocol nonblocking }; %CompatibilityMethods = ( ip => 'host', ispublic => 'public', ); ############################################################################### ### M E T H O D S ############################################################################### ### (CONSTRUCTOR) METHOD: new( %config ) ### Create and initialize a new icecast server connection. The configuration ### hash is in the following form: ### ### ( ### host => , ### port => , ### mount => , ### nonblocking => , ### password => , ### user => , ### dumpfile => , ### name => , ### url => , ### genre => , ### format => , ### protocol => , ### description => , ### public => ### ) ### ### None of the keys are mandatory, and may be set after the connection object ### is created. This method returns the initialized icecast server ### connection object. Returns the undefined value on failure. sub new { my $proto = shift; my $class = ref $proto || $proto; my ( %args, # The config pseudo-hash $self, # The shout_conn_t object ); ### Unwrap the pseudo-hash into a real one %args = @_; ### Call our parent's constructor $self = $class->raw_new() or return undef; ### Set each of the config hash elements by using the keys of the ### config pseudo-hash as the method name foreach my $method ( keys %args ) { ### Allow keys to be of varying case and preceeded by an optional '-' $method =~ s{^-}{}; $method = lc $method; ### Turn off strict references so we can use a variable as a method name NO_STRICT: { no strict 'refs'; $self->$method( $args{$method} ); } } return $self; } ### METHOD: open( undef ) ### Connect to the target server. Returns undef and sets the object error ### message if the open fails; returns a true value if the open ### succeeds. sub open { my $self = shift or croak "open: Method called as function"; $self->shout_open ? 0 : 1; } ### METHOD: close( undef ) ### Disconnect from the target server. Returns undef and sets the object error ### message if the close fails; returns a true value if the close ### succeeds. sub close { my $self = shift or croak "close: Method called as function"; $self->shout_close ? 0 : 1; } ### METHOD: get_errno( undef ) ### Returns a machine-readable integer if one has occurred in the ### object. Returns the undefined value if no error has occurred. sub get_errno { my $self = shift or croak "get_errno: Method called as function"; $self->shout_get_errno or undef; } ### METHOD: get_error( undef ) ### Returns a human-readable get_error message if one has occurred in the ### object. Returns the undefined value if no error has occurred. sub get_error { my $self = shift or croak "get_error: Method called as function"; $self->shout_get_error or undef; } ### METHOD: set_metadata(key => value,key => value,...) ### Sets the metadata for the connection. Returns a true value if the update ### succeeds, and the undefined value if it fails. sub set_metadata ($$) { my $self = shift or croak "set_metadata: Method called as function"; my %param=@_; my $md=shout_metadata_new(); for my $k (keys %param) { shout_metadata_add($md,$k,$param{$k}); } $self->shout_set_metadata($md) ? 0 : 1; } ### METHOD: send( $data[, $length] ) ### Send the specified data with the optional length to the Icecast ### server. Returns a true value on success, and returns the undefined value ### after setting the per-object error message on failure. sub send ($$) { my $self = shift or croak "send_data: Method called as function"; my $data = shift or croak "send_data: No data specified"; my $len = shift || length $data; $self->shout_send( $data, $len ) ? 0 : 1; } ### METHOD: sync( undef ) ### Sleep until the connection is ready for more data. This function should be ### used only in conjuction with C, in order to send data ### at the correct speed to the icecast server. sub sync ($) { my $self = shift or croak "sync: Method called as function"; $self->shout_sync; } ### METHOD: delay( undef ) ### Tell how much time (in seconds and fraction of seconds) must be ### waited until more data can be sent. Use instead of sync() to ### allow you to do other things while waiting. ### Used only in conjuction with C, in order to send data ### at the correct speed to the icecast server. sub delay ($) { my $self = shift or croak "delay: Method called as function"; $self->shout_delay; } ### METHOD: queuelen( undef ) ### Tell how many bytes of data are in the write queue. This is only ### useful in conjunction with nonblocking sends. sub queuelen ($) { my $self = shift or croak "delay: Method called as function"; $self->shout_queuelen; } ### METHOD: set_audio_info( key => val, key => val ) ### Set audio parameters (bitrate, samplerate, channels etc) for informational ### purposes. Audio info is a hash using the SHOUT_AI constants as keys. sub set_audio_info ($$) { my $self = shift or croak "set_audio_info: Method called as function"; my %param=@_; for my $k (keys %param) { $self->shout_set_audio_info($k, $param{$k}) and return 0; } 1; } ### METHOD: get_audio_info( key ) ### Returns audio parameters sub get_audio_info ($$) { my $self = shift or croak "get_audio_info: Method called as function"; my $k = shift or croak "get_audio_info: No parameter supplied"; return $self->shout_get_audio_info($k); } ### COMPATIBILITY with Shout 1.0 *Shout::disconnect = *Shout::close; *Shout::sendData = *Shout::send; *Shout::error = *Shout::get_error; *Shout::sleep = *Shout::sync; ### Compatibility method: connect ( undef ) ### Sets format to MP3, then calls open sub connect ($) { my $self = shift or croak "connect: Method called as function"; $self->format(Shout::SHOUT_FORMAT_MP3()); return $self->open(); } ### Compatibility method: icy_compat ( $compat ) ### Translates the icy_compat call to set protocol to icy or xaudiocast sub icy_compat ($$) { my $self = shift or croak "icy_compat: Method called as function"; if (@_) { my $compat = shift; if ($compat) { $self->protocol(Shout::SHOUT_PROTOCOL_ICY()) ? 0 : 1; } else { $self->protocol(Shout::SHOUT_PROTOCOL_XAUDIOCAST()) ? 0 : 1; } } else { return ($self->protocol == Shout::SHOUT_PROTOCOL_ICY()) ? 1 : 0; } } ### Compatibility method: bitrate ( $bitrate ) ### Translates the bitrate call to the appropriate audio_info call sub bitrate ($$) { my $self = shift or croak "bitrate: Method called as function"; if (@_) { my $br = shift or croak "bitrate: No parameter specified"; $self->set_audio_info(Shout::SHOUT_AI_BITRATE(), $br) ? 0 : 1; } else { return $self->get_audio_info(Shout::SHOUT_AI_BITRATE()); } } ### Compatibility method: updateMetadata ( $metadata) ### Translates the metadata call to new form sub updateMetadata ($$) { my $self = shift or croak "updateMetadata: Method called as function"; my $metadata = shift or croak "updateMetadata: No metadata specified"; return $self->set_metadata(song => $metadata) ? 0 : 1; } ############################################################################### ### A U T O L O A D E D M E T H O D S ############################################################################### ### METHOD: port( $portNumber ) ### Get/set the port to connect to on the target Icecast server. ### METHOD: mount( $mountPointName ) ### Get/set the mountpoint to use when connecting to the server. ### METHOD: password( $password ) ### Get/set the password to use when connecting to the Icecast server. ### METHOD: user( $username ) ### Get/set the username to use when connecting to the Icecast server. ### METHOD: dumpfile( $filename ) ### Get/set the name of the icecast dumpfile for the stream. The dumpfile is a ### special feature of recent icecast servers. When dumpfile is not ### undefined, and the x-audiocast protocol is being used, the icecast ### server will save the stream locally to a dumpfile (a dumpfile is just a ### raw mp3 stream dump). Using this feature will cause data to be written ### to the drive on the icecast server, so use with caution, or you will ### fill up your disk! ### METHOD: name( $nameString ) ### Get/set the name of the stream. ### METHOD: url( $urlString ) ### Get/set the url of the stream's homepage. ### METHOD: genre( $genreString ) ### Get/set the stream's genre. ### METHOD: description( $descriptionString ) ### Get/set the description of the stream. ### METHOD: public( $boolean ) ### Get/set the connection's public flag. This flag, when set to true, indicates ### that the stream may be listed in the public directory servers. ### METHOD: nonblocking( $boolean ) ### Get/set the connection's nonblocking flag. This flag, when set to true, indicates ### that sends should complete instantly, queueing data if necessary. ### (PROXY) METHOD: AUTOLOAD( @args ) ### Provides a proxy for functions and methods which aren't explicitly defined. sub AUTOLOAD { ( my $method = $AUTOLOAD ) =~ s/.*:://; # Translate Shout 1.0 calls. This should probably be made optional. if ( defined($CompatibilityMethods{$method}) ) { $method = $CompatibilityMethods{$method}; } if (grep {$_ eq $method} @TranslatedMethods) { NOSTRICT: { no strict 'refs'; my $setMethod = "shout_set_$method"; my $getMethod = "shout_get_$method"; *$AUTOLOAD = sub ($$) { my $obj = shift; return $obj->$setMethod(@_) if @_; return $obj->$getMethod(); }; } goto &$AUTOLOAD; } # Check for string or integer constants my $val = strconstant($method, @_ ? $_[0] : 0); if ($! != 0 && ($! =~ /Invalid/ || $!{EINVAL})) { $val = constant($method, @_ ? $_[0] : 0); } if ($! == 0) { NOSTRICT: { no strict 'refs'; *$method = sub { $val }; goto &$method; } } croak "No such Shout constant '$method'"; } ### Module return value indicates successful loading 1; __END__ ### AUTOGENERATED DOCUMENTATION FOLLOWS =head1 METHODS =over 4 =item I Connect to the target server. Returns undef and sets the object error message if the open fails; returns a true value if the open succeeds. =item I Get/set the description of the stream. =item I Disconnect from the target server. Returns undef and sets the object error message if the close fails; returns a true value if the close succeeds. =item I Get/set the name of the icecast dumpfile for the stream. The dumpfile is a special feature of recent icecast servers. When dumpfile is not undefined, and the x-audiocast protocol is being used, the icecast server will save the stream locally to a dumpfile (a dumpfile is just a raw mp3 stream dump). Using this feature will cause data to be written to the drive on the icecast server, so use with caution, or you will fill up your disk! =item I Returns a human-readable error message if one has occurred in the object. Returns the undefined value if no error has occurred. =item I Returns a machine-readable error integer if an error has occurred in the object. Returns the undefined value if no error has occurred. =item I Get/set the stream's genre. =item I Get/set the target host for the connection. Argument can be either an address or a hostname. It is a fatal error is the argument is a hostname, and the numeric address cannot be resolved from it. =item I Get/set the mountpoint to use when connecting to the server. =item I Get/set the name of the stream. =item I Get/set the password to use when connecting to the Icecast server. =item I Get/set the username to use when connecting to the Icecast server. =item I Get/set the port to connect to on the target Icecast server. =item I Get/set the connection's public flag. This flag, when set to true, indicates that the stream may be listed in the public directory servers. =item I Send the specified data with the optional length to the Icecast server. Returns a true value on success, and returns the undefined value after setting the per-object error message on failure. =item I Sleep until the connection is ready for more data. This function should be used only in conjuction with C, in order to send data at the correct speed to the icecast server. =item I Tell how much time (in seconds and fraction of seconds) must be waited until more data can be sent. Use instead of sync() to allow you to do other things while waiting. =item I Update the metadata for the connection. Returns a true value if the update succeeds, and the undefined value if it fails. =item I Get/set the url of the stream's homepage. =back =head2 Constructor Methods =over 4 =item I Create and initialize a new icecast server connection. The configuration hash is in the following form: ( host => , port => , mount => , password => , user => , dumpfile => , name => , url => , genre => , description => , public => , ) None of the keys are mandatory, and may be set after the connection object is created. This method returns the initialized icecast server connection object. Returns the undefined value on failure. =back =head2 Proxy Methods =over 4 =item I Provides a proxy for functions and methods which aren't explicitly defined. =back =cut