diff options
-rwxr-xr-x | docs/CHANGELOG | 2 | ||||
-rw-r--r-- | lib/class/stream.class.php | 464 | ||||
-rw-r--r-- | lib/class/stream_playlist.class.php | 451 | ||||
-rw-r--r-- | lib/class/update.class.php | 25 | ||||
-rw-r--r-- | modules/flash/xspf_player.php | 5 | ||||
-rw-r--r-- | play/index.php | 23 | ||||
-rw-r--r-- | stream.php | 18 |
7 files changed, 522 insertions, 466 deletions
diff --git a/docs/CHANGELOG b/docs/CHANGELOG index 1b7ff2d4..a839d76a 100755 --- a/docs/CHANGELOG +++ b/docs/CHANGELOG @@ -4,6 +4,8 @@ -------------------------------------------------------------------------- v.3.6-FUTURE + - Fixed streaming on Android devices and anything else that expects to + be able to pass a playlist URL to an application and have it work - Removed the SHOUTcast localplay controller -------------------------------------------------------------------------- diff --git a/lib/class/stream.class.php b/lib/class/stream.class.php index adcfe87b..293e0f88 100644 --- a/lib/class/stream.class.php +++ b/lib/class/stream.class.php @@ -1,8 +1,6 @@ <?php /* vim:set tabstop=8 softtabstop=8 shiftwidth=8 noexpandtab: */ /** - * Stream Class - * * * LICENSE: GNU General Public License, version 2 (GPLv2) * Copyright (c) 2001 - 2011 Ampache.org All Rights Reserved @@ -20,97 +18,17 @@ * along with this program; if not, write to the Free Software * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. * - * @package Ampache - * @copyright 2001 - 2011 Ampache.org - * @license http://opensource.org/licenses/gpl-2.0 GPLv2 - * @link http://www.ampache.org/ */ -/** - * Stream Class - * - * This class is used to generate the Playlists and pass them on - * With Localplay this actually just sends the commands to the localplay - * module in question. It has two sources for data - * songs (array of ids) and urls (array of full urls) - * - * @package Ampache - * @copyright 2001 - 2011 Ampache.org - * @license http://opensource.org/licenses/gpl-2.0 GPLv2 - * @link http://www.ampache.org/ - */ -class Stream { - /* Variables from DB */ - public $type; - public $media = array(); - public $urls = array(); - public $user_id; +class Stream { - // Generate once an object is constructed public static $session; - - // Let's us tell if the session has been activated private static $session_inserted; - /** - * Constructor for the stream class takes a type and an array - * of song ids - */ - public function __construct($type='m3u', $media_ids) { - - $this->type = $type; - $this->media = $media_ids; - $this->user_id = $GLOBALS['user']->id; - - if (!is_array($this->media)) { settype($this->media,'array'); } - - } // Constructor - - /** - * start - *runs this and depending on the type passed it will - *call the correct function - */ - public function start() { - - if (!count($this->media) AND !count($this->urls)) { - debug_event('stream','Error: No Songs Passed on ' . $this->type . ' stream','2'); - return false; - } - - // We're starting insert the session into session_stream - if (!self::get_session()) { - debug_event('stream','Session Insertion failure, aborting','3'); - return false; - } - - $methods = get_class_methods('Stream'); - $create_function = "create_" . $this->type; - - // If in the class, call it - if (in_array($create_function,$methods)) { - $this->{$create_function}(); - } - // Assume M3u incase they've pooched the type - else { - $this->create_m3u(); - } - - } // start - - /** - * add_urls - * Add an array of urls, it may be a single one who knows, this - * is used for things that aren't coming from media objects - */ - public function add_urls($urls=array()) { - - if (!is_array($urls)) { return false; } - - $this->urls = array_merge($urls,$this->urls); - - } // manual_url_add + private function __construct() { + // Static class, do nothing. + } /** * get_session @@ -184,8 +102,10 @@ class Stream { /** * gc_session - * This function performes the garbage collection stuff, run on extend and on now playing refresh - * There is an array of agents that we will never GC because of their nature, MPD being the best example + * This function performes the garbage collection stuff, run on extend + * and on now playing refresh. + * There is an array of agents that we will never GC because of their + * nature, MPD being the best example. */ public static function gc_session($ip='',$agent='',$uid='',$sid='') { @@ -195,6 +115,8 @@ class Stream { $sql = "DELETE FROM `session_stream` WHERE `expire` < '$time'"; $db_results = Dba::write($sql); + Stream_Playlist::clean(); + foreach ($append_array as $append_agent) { if (strpos(strtoupper($agent), $append_agent) !== false) { // We're done here jump ship! @@ -234,372 +156,6 @@ class Stream { } // extend_session /** - * create_simplem3u - * this creates a simple m3u without any of the extended information - */ - public function create_simple_m3u() { - - header("Cache-control: public"); - header("Content-Disposition: filename=ampache_playlist.m3u"); - header("Content-Type: audio/x-mpegurl;"); - - // Flip for the poping! - asort($this->urls); - - /* Foreach songs */ - foreach ($this->media as $element) { - $type = array_shift($element); - echo call_user_func(array($type,'play_url'),array_shift($element)) . "\n"; - } // end foreach - - /* Foreach the additional URLs */ - foreach ($this->urls as $url) { - echo "$url\n"; - } - - } // simple_m3u - - /** - * create_m3u - * creates an m3u file, this includes the EXTINFO and as such can be - * large with very long playlsits - */ - public function create_m3u() { - - // Send the client an m3u playlist - header("Cache-control: public"); - header("Content-Disposition: filename=ampache_playlist.m3u"); - header("Content-Type: audio/x-mpegurl;"); - echo "#EXTM3U\n"; - - // Foreach the songs in this stream object - foreach ($this->media as $element) { - $type = array_shift($element); - $media = new $type(array_shift($element)); - $media->format(); - switch ($type) { - case 'song': - echo "#EXTINF:$media->time," . $media->f_artist_full . " - " . $media->title . "\n"; - break; - case 'video': - echo "#EXTINF: Video - $media->title\n"; - break; - case 'radio': - echo "#EXTINF: Radio - $media->name [$media->frequency] ($media->site_url)\n"; - break; - case 'random': - echo "#EXTINF:Random URL\n"; - break; - default: - echo "#EXTINF:URL-Add\n"; - break; - } - echo call_user_func(array($type,'play_url'),$media->id) . "\n"; - } // end foreach - - /* Foreach URLS */ - foreach ($this->urls as $url) { - echo "#EXTINF: URL-Add\n"; - echo $url . "\n"; - } - - } // create_m3u - - /** - * create_pls - * This creates a new pls file from an array of songs and - * urls, exciting I know - */ - public function create_pls() { - - /* Count entries */ - $total_entries = count($this->media) + count($this->urls); - - // Send the client a pls playlist - header("Cache-control: public"); - header("Content-Disposition: filename=ampache_playlist.pls"); - header("Content-Type: audio/x-scpls;"); - echo "[playlist]\n"; - echo "NumberOfEntries=$total_entries\n"; - foreach ($this->media as $element) { - $i++; - $type = array_shift($element); - $media = new $type(array_shift($element)); - $media->format(); - switch ($type) { - case 'song': - $name = $media->f_artist_full . " - " . $media->title . "." . $media->type; - $length = $media->time; - break; - default: - $name = 'URL-Add'; - $length='-1'; - break; - } - - $url = call_user_func(array($type,'play_url'),$media->id); - echo "File" . $i . "=$url\n"; - echo "Title" . $i . "=$name\n"; - echo "Length" . $i . "=$length\n"; - } // end foreach songs - - /* Foreach Additional URLs */ - foreach ($this->urls as $url) { - $i++; - echo "File" . $i ."=$url\n"; - echo "Title". $i . "=AddedURL\n"; - echo "Length" . $i . "=-1\n"; - } // end foreach urls - - echo "Version=2\n"; - - } // create_pls - - /** - * create_asx - * creates an ASX playlist (Thx Samir Kuthiala) This should really only be used - * if all of the content is ASF files. - */ - public function create_asx() { - - header("Cache-control: public"); - header("Content-Disposition: filename=ampache_playlist.asx"); - header("Content-Type: video/x-ms-wmv;"); - - echo "<ASX version = \"3.0\" BANNERBAR=\"AUTO\">\n"; - echo "<TITLE>Ampache ASX Playlist</TITLE>"; - - foreach ($this->media as $element) { - $type = array_shift($element); - $media = new $type(array_shift($element)); - $media->format(); - switch ($type) { - case 'song': - $name = $media->f_album_full . " - " . $media->title . "." . $media->type; - $author = $media->f_artist_full; - break; - default: - $author = 'Ampache'; - $name = 'URL-Add'; - break; - } // end switch - $url = call_user_func(array($type,'play_url'),$media->id); - - echo "<ENTRY>\n"; - echo "<TITLE>$name</TITLE>\n"; - echo "<AUTHOR>$author</AUTHOR>\n"; - echo "\t\t<COPYRIGHT>".$media->year."</COPYRIGHT>\n"; - echo "\t\t<DURATION VALUE=\"00:00:".$media->time."\" />\n"; - echo "\t\t<PARAM NAME=\"Album\" Value=\"".$media->f_album_full."\" />\n"; - echo "\t\t<PARAM NAME=\"Genre\" Value=\"".$media->get_genre_name()."\" />\n"; - echo "\t\t<PARAM NAME=\"Composer\" Value=\"".$media->f_artist_full."\" />\n"; - echo "\t\t<PARAM NAME=\"Prebuffer\" Value=\"false\" />\n"; - echo "<REF HREF = \"". $url . "\" />\n"; - echo "</ENTRY>\n"; - - } // end foreach - - /* Foreach urls */ - foreach ($this->urls as $url) { - echo "<ENTRY>\n"; - echo "<TITLE>AddURL</TITLE>\n"; - echo "<AUTHOR>AddURL</AUTHOR>\n"; - echo "<REF HREF=\"$url\" />\n"; - echo "</ENTRY>\n"; - } // end foreach - - echo "</ASX>\n"; - - } // create_asx - - /** - * create_xspf - * creates an XSPF playlist (Thx PB1DFT) - */ - public function create_xspf() { - - // Itterate through the songs - foreach ($this->media as $element) { - $type = array_shift($element); - $media = new $type(array_shift($element)); - $media->format(); - - $xml = array(); - - switch ($type) { - default: - case 'song': - $xml['track']['title'] = $media->title; - $xml['track']['creator'] = $media->f_artist_full; - $xml['track']['info'] = Config::get('web_path') . "/albums.php?action=show&album=" . $media->album; - $xml['track']['image'] = Config::get('web_path') . "/image.php?id=" . $media->album . "&thumb=3"; - $xml['track']['album'] = $media->f_album_full; - $length = $media->time; - break; - case 'video': - $xml['track']['title'] = $media->title; - $xml['track']['creator'] = $media->f_artist_full; - $xml['track']['info'] = Config::get('web_path') . '/browse.php?action=video'; - $xml['track']['image'] = Config::get('web_path') . '/image.php?id=' . $media->id . '&type=video&thumb=3&sid=' . session_id(); - $xml['track']['meta'] = array('attribute'=>'rel="provider"','value'=>'video'); - break; - } // type - - $xml['track']['location'] = call_user_func(array($type,'play_url'),$media->id); - $xml['track']['identifier'] = $xml['track']['location']; - $xml['track']['duration'] = $length * 1000; - - $result .= xmlData::keyed_array($xml,1); - - } // end foreach - - xmlData::set_type('xspf'); - - header("Cache-control: public"); - header("Content-Disposition: filename=ampache_playlist.xspf"); - header("Content-Type: application/xspf+xml; charset=utf-8"); - echo xmlData::header(); - echo $result; - echo xmlData::footer(); - - } // create_xspf - - /** - * create_xspf_player - * due to the fact that this is an integrated player (flash) we actually - * have to do a little 'cheating' to make this work, we are going to take - * advantage of tmp_playlists to do all of this hotness - */ - public function create_xspf_player() { - - /* Build the extra info we need to have it pass */ - $play_info = "?action=show&tmpplaylist_id=" . $GLOBALS['user']->playlist->id; - - // start ugly evil javascript code - //FIXME: This needs to go in a template, here for now though - //FIXME: This preference doesn't even exists, we'll eventually - //FIXME: just make it the default - if (Config::get('embed_xspf') == 1 ){ - header("Location: ".Config::get('web_path')."/index.php?xspf&play_info=".$GLOBALS['user']->playlist->id); - } - else { - echo "<html><head>\n"; - echo "<title>" . Config::get('site_title') . "</title>\n"; - echo "<script language=\"javascript\" type=\"text/javascript\">\n"; - echo "<!-- begin\n"; - echo "function PlayerPopUp(URL) {\n"; - // We do a little check here to see if it's a Wii! - if (false !== stristr($_SERVER['HTTP_USER_AGENT'], 'Nintendo Wii')) { - echo "window.location=URL;\n"; - } - // Else go ahead and do the normal stuff - else { - echo "window.open(URL, 'XSPF_player', 'width=400,height=170,scrollbars=0,toolbar=0,location=0,directories=0,status=0,resizable=0');\n"; - echo "window.location = '" . return_referer() . "';\n"; - echo "return false;\n"; - } - echo "}\n"; - echo "// end -->\n"; - echo "</script>\n"; - echo "</head>\n"; - - echo "<body onLoad=\"javascript:PlayerPopUp('" . Config::get('web_path') . "/modules/flash/xspf_player.php" . $play_info . "')\">\n"; - echo "</body>\n"; - echo "</html>\n"; - } - } // create_xspf_player - - /** - * create_localplay - * This calls the Localplay API and attempts to - * add, and then start playback - */ - public function create_localplay() { - - // First figure out what their current one is and create the object - $localplay = new Localplay(Config::get('localplay_controller')); - $localplay->connect(); - foreach ($this->media as $element) { - $type = array_shift($element); - switch ($type) { - case 'video': - // Add check for video support - case 'song': - case 'radio': - case 'random': - $media = new $type(array_shift($element)); - break; - default: - $media = array_shift($element); - break; - } // switch on types - $localplay->add($media); - } // foreach object - - /** - * Add urls after the fact - */ - foreach ($this->urls as $url) { - $localplay->add($url); - } - - $localplay->play(); - - } // create_localplay - - /** - * create_democratic - * This 'votes' on the songs it inserts them into - * a tmp_playlist with user of -1 (System) - */ - public function create_democratic() { - - $democratic = Democratic::get_current_playlist(); - $democratic->set_parent(); - $democratic->add_vote($this->media); - - } // create_democratic - - /** - * create_download - * This prompts for a download of the song, only a single - * element can by in song_ids - */ - private function create_download() { - - // There should only be one here... - foreach ($this->media as $element) { - $type = array_shift($element); - $media = new $type(array_shift($element)); - $url = call_user_func(array($type,'play_url'),$media->id); - - // Append the fact we are downloading - $url .= '&action=download'; - - // Header redirect baby! - header("Location: $url"); - exit; - } - - } //create_download - - /** - * create_ram - *this functions creates a RAM file for use by Real Player - */ - public function create_ram() { - - header("Cache-control: public"); - header("Content-Disposition: filename=ampache_playlist.ram"); - header("Content-Type: audio/x-pn-realaudio ram;"); - foreach ($this->media as $element) { - $type = array_shift($element); - echo $url = call_user_func(array($type,'play_url'),array_shift($element)) . "\n"; - } // foreach songs - - } // create_ram - - /** * start_transcode * * This is a rather complex function that starts the transcoding or diff --git a/lib/class/stream_playlist.class.php b/lib/class/stream_playlist.class.php new file mode 100644 index 00000000..1a34f0b9 --- /dev/null +++ b/lib/class/stream_playlist.class.php @@ -0,0 +1,451 @@ +<?php +/* vim:set tabstop=8 softtabstop=8 shiftwidth=8 noexpandtab: */ +/** + * + * LICENSE: GNU General Public License, version 2 (GPLv2) + * Copyright (c) 2001 - 2011 Ampache.org All Rights Reserved + * + * This program is free software; you can redistribute it and/or + * modify it under the terms of the GNU General Public License v2 + * as published by the Free Software Foundation. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program; if not, write to the Free Software + * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. + * + */ + +/** + * Stream_Playlist Class + * + * This class is used to generate the Playlists and pass them on + * With Localplay this actually just sends the commands to the localplay + * module in question. + */ + +class Stream_Playlist { + + public $id; + public $urls = array(); + public $user; + + /** + * Stream_Playlist constructor + * If an ID is passed, it should be a stream session ID. + */ + public function __construct($id = null) { + + if($id) { + Stream::set_session($id); + } + + $this->id = Dba::escape(Stream::get_session()); + + if (!Stream::session_exists($this->id)) { + debug_event('stream_playlist', 'Stream::session_exists failed', 2); + return false; + } + + $this->user = intval($GLOBALS['user']->id); + + $sql = "SELECT * FROM `stream_playlist` WHERE `sid`='" . + $this->id . "' ORDER BY `id`"; + + $db_results = Dba::read($sql); + + while ($row = Dba::fetch_assoc($db_results)) { + $this->urls[] = new Stream_URL($row); + } + + return true; + } + + private function _add_url($url) { + $this->urls[] = $url; + + $sql = 'INSERT INTO `stream_playlist` '; + + $fields[] = '`sid`'; + $values[] = Dba::escape($this->id); + + foreach ($url->properties as $field) { + if ($url->$field) { + $fields[] = '`' . $field . '`'; + $values[] = Dba::escape($url->$field); + } + } + $sql .= '(' . implode(', ', $fields) . ') '; + $sql .= "VALUES('" . implode("', '", $values) . "')"; + + return Dba::write($sql); + } + + public static function clean() { + $sql = 'DELETE FROM `stream_playlist` ' . + 'USING `stream_playlist` LEFT JOIN `session_stream` ' . + 'ON `session_stream`.`id`=`stream_playlist`.`sid` ' . + 'WHERE `session_stream`.`id` IS NULL'; + return Dba::write($sql); + } + + /** + * _media_to_urlarray + * Formats the URL and media information and adds it to the object + */ + private static function _media_to_urlarray($media) { + $urls = array(); + foreach($media as $medium) { + debug_event('stream_playlist', 'Adding ' . json_encode($media), 5); + $url = array(); + + $type = $medium['object_type']; + $array['type'] = $type; + + $object = new $type($medium['object_id']); + $object->format(); + //FIXME: play_url shouldn't be static + $url['url'] = $type::play_url($object->id); + + // Set a default which can be overridden + $url['author'] = 'Ampache'; + $url['time'] = $object->time; + switch($type) { + case 'song': + $url['title'] = $object->title; + $url['author'] = $object->f_artist_full; + $url['info_url'] = $object->f_link; + $url['image_url'] = Art::url($object->album, 'album'); + $url['album'] = $object->f_album_full; + break; + case 'video': + $url['title'] = 'Video - ' . $object->title; + $url['author'] = $object->f_artist_full; + break; + case 'radio': + $url['title'] = 'Radio - ' . $object->name . + ' [' . $object->frequency . + '] (' . $object->site_url . ')'; + break; + case 'random': + $url['title'] = 'Random URL'; + break; + default: + $url['title'] = 'URL-Add'; + $url['time'] = -1; + break; + } + + $urls[] = new Stream_URL($url); + } + + return $urls; + } + + public function generate_playlist($type, $redirect = false) { + + if (!count($this->urls)) { + debug_event('stream_playlist', 'Error: Empty URL array for ' . $this->id, 2); + return false; + } + + debug_event('stream_playlist', 'generating a ' . $type, 5); + + $ext = $type; + switch($type) { + case 'democratic': + case 'localplay': + case 'xspf_player': + // These are valid, but witchy + $redirect = false; + unset($ext); + break; + case 'asx': + $ct = 'video/x-ms-wmv'; + break; + case 'pls': + $ct = 'audio/x-scpls'; + break; + case 'ram': + $ct = 'audio/x-pn-realaudio ram'; + break; + case 'simple_m3u': + $ext = 'm3u'; + $ct = 'audio/x-mpegurl'; + break; + case 'xspf': + $ct = 'application/xspf+xml'; + break; + case 'm3u': + default: + // Assume M3U if the pooch is screwed + $ext = $type = 'm3u'; + $ct = 'audio/x-mpegurl'; + break; + } + + if ($redirect) { + // Our ID is the SID, so we always want to include it + Config::set('require_session', true, true); + header('Location: ' . Stream::get_base_url() . 'uid=' . scrub_out($this->user) . '&type=playlist&playlist_type=' . scrub_out($type)); + exit; + } + + if (isset($ext)) { + header('Cache-control: public'); + header('Content-Disposition: filename=ampache_playlist.' . $ext); + header('Content-Type: ' . $ct . ';'); + } + + $this->{'create_' . $type}(); + } + + /** + * add + * Adds an array of media + */ + public function add($media = array()) { + $urls = $this->_media_to_urlarray($media); + foreach ($urls as $url) { + $this->_add_url($url); + } + } + + /** + * add_urls + * Add an array of urls. This is used for things that aren't coming + * from media objects + */ + public function add_urls($urls = array()) { + + if (!is_array($urls)) { return false; } + + foreach ($urls as $url) { + $this->_add_url(new Stream_URL(array( + 'url' => $url, + 'title' => 'URL-Add', + 'author' => 'Ampache', + 'time' => '-1' + ))); + } + } + + /** + * create_simplem3u + * this creates a simple m3u without any of the extended information + */ + public function create_simple_m3u() { + + foreach ($this->urls as $url) { + echo $url->url . "\n"; + } + + } // simple_m3u + + /** + * create_m3u + * creates an m3u file, this includes the EXTINFO and as such can be + * large with very long playlsits + */ + public function create_m3u() { + + echo "#EXTM3U\n"; + + foreach ($this->urls as $url) { + echo '#EXTINF:' . $url->time, ',' . $url->author . + ' - ' . $url->title . "\n"; + echo $url->url . "\n"; + } + + } // create_m3u + + /** + * create_pls + */ + public function create_pls() { + + echo "[playlist]\n"; + echo 'NumberOfEntries=' . count($this->urls) . "\n"; + foreach ($this->urls as $url) { + $i++; + echo 'File' . $i . '='. $url->url . "\n"; + echo 'Title' . $i . '=' . $url->author . ' - ' . + $url->title . "\n"; + echo 'Length' . $i . '=' . $url->time . "\n"; + } + + echo "Version=2\n"; + } // create_pls + + /** + * create_asx + * This should really only be used if all of the content is ASF files. + */ + public function create_asx() { + + echo '<ASX version = "3.0" BANNERBAR="AUTO">' . "\n"; + echo "<TITLE>Ampache ASX Playlist</TITLE>"; + + foreach ($this->urls as $url) { + echo "<ENTRY>\n"; + echo '<TITLE>' . $url->title . "</TITLE>\n"; + echo '<AUTHOR>' . $url->author . "</AUTHOR>\n"; + echo "\t\t" . '<DURATION VALUE="00:00:' . $url->time . '" />' . "\n"; + echo "\t\t" . '<PARAM NAME="Album" Value="' . $url->album . '" />' . "\n"; + echo "\t\t" . '<PARAM NAME="Composer" Value="' . $url->author . '" />' . "\n"; + echo "\t\t" . '<PARAM NAME="Prebuffer" Value="false" />' . "\n"; + echo '<REF HREF = "' . $url->url . '" />' . "\n"; + echo "</ENTRY>\n"; + } + + echo "</ASX>\n"; + + } // create_asx + + /** + * create_xspf + */ + public function create_xspf() { + + foreach ($this->urls as $url) { + $xml = array(); + + $xml['track'] = array( + 'title' => $url->title, + 'creator' => $url->author, + // FIXME: regression + // video: ['meta'] = array('attribute'=>'rel="provider"','value'=>'video') + 'duration' => $url->time * 1000, + 'location' => $url->url, + 'identifier' => $url->url + ); + if ($url->info_url) { + $xml['track']['info'] = $url->info_url; + } + if ($url->image_url) { + $xml['track']['image'] = $url->image_url; + } + if ($url->album) { + $xml['track']['album'] = $url->album; + } + + $result .= xmlData::keyed_array($xml, true); + + } // end foreach + + xmlData::set_type('xspf'); + echo xmlData::header(); + echo $result; + echo xmlData::footer(); + + } // create_xspf + + /** + * create_xspf_player + * Due to the fact that this is an integrated player (flash) we actually + * have to do a little 'cheating' to make this work. + * We are going to take advantage of tmp_playlists to do all of this + * hotness + */ + public function create_xspf_player() { + debug_event('stream_playlist', 'Creating XSPF player', 5); + /* Build the extra info we need to have it pass */ + $play_info = "?action=show&tmpplaylist_id=" . $GLOBALS['user']->playlist->id; + + // start ugly evil javascript code + //FIXME: This needs to go in a template, here for now though + //FIXME: This preference doesn't even exists, we'll eventually + //FIXME: just make it the default + if (Config::get('embed_xspf') == 1 ){ + header("Location: ".Config::get('web_path')."/index.php?xspf&play_info=".$GLOBALS['user']->playlist->id); + } + else { + echo "<html><head>\n"; + echo "<title>" . Config::get('site_title') . "</title>\n"; + echo "<script language=\"javascript\" type=\"text/javascript\">\n"; + echo "<!-- begin\n"; + echo "function PlayerPopUp(URL) {\n"; + // We do a little check here to see if it's a Wii! + if (false !== stristr($_SERVER['HTTP_USER_AGENT'], 'Nintendo Wii')) { + echo "window.location=URL;\n"; + } + // Else go ahead and do the normal stuff + else { + echo "window.open(URL, 'XSPF_player', 'width=400,height=170,scrollbars=0,toolbar=0,location=0,directories=0,status=0,resizable=0');\n"; + echo "window.location = '" . return_referer() . "';\n"; + echo "return false;\n"; + } + echo "}\n"; + echo "// end -->\n"; + echo "</script>\n"; + echo "</head>\n"; + + echo "<body onLoad=\"javascript:PlayerPopUp('" . Config::get('web_path') . "/modules/flash/xspf_player.php" . $play_info . "')\">\n"; + echo "</body>\n"; + echo "</html>\n"; + } + } // create_xspf_player + + /** + * create_localplay + * This calls the Localplay API to add the URLs and then start playback + */ + public function create_localplay() { + + $localplay = new Localplay(Config::get('localplay_controller')); + $localplay->connect(); + foreach ($this->urls as $url) { + $localplay->add_url($url); + } + + $localplay->play(); + + } // create_localplay + + /** + * create_democratic + * This 'votes' on the songs it inserts them into + * a tmp_playlist with user of -1 (System) + */ + public function create_democratic() { + + $democratic = Democratic::get_current_playlist(); + $democratic->set_parent(); + $democratic->add_vote($this->media); + + } // create_democratic + + /** + * create_download + * This prompts for a download of the song + */ + private function create_download() { + + // There should only be one here... + if (count($this->urls) != 1) { + debug_event('stream_playlist', 'Download called, but $urls contains ' . json_encode($this->urls), 2); + } + + // Header redirect baby! + $url = current($this->urls); + header('Location: ' . $url->url . '&action=download'); + exit; + } //create_download + + /** + * create_ram + *this functions creates a RAM file for use by Real Player + */ + public function create_ram() { + foreach ($this->urls as $url) { + echo $url->url . "\n"; + } + } // create_ram + +} + +?> diff --git a/lib/class/update.class.php b/lib/class/update.class.php index 31446e76..7a32c776 100644 --- a/lib/class/update.class.php +++ b/lib/class/update.class.php @@ -377,6 +377,9 @@ class Update { $update_string = '- Allow compound MBIDs in the artist table.<br />'; $version[] = array('version' => '360010', 'description' => $update_string); + $update_string = '- Add table to store stream session playlist.<br />'; + $version[] = array('version' => '360011', 'description' => $update_string); + return $version; } // populate_version @@ -2089,5 +2092,27 @@ class Update { self::set_version('db_version', '360010'); } + /** + * update_380011 + * We need a place to store actual playlist data for downloadable + * playlist files. + */ + public static function update_360011() { + $sql = 'CREATE TABLE `stream_playlist` (' . + '`id` int(11) unsigned NOT NULL AUTO_INCREMENT,' . + '`sid` varchar(64) COLLATE utf8_unicode_ci NOT NULL,' . + '`url` text COLLATE utf8_unicode_ci NOT NULL,' . + '`info_url` text COLLATE utf8_unicode_ci DEFAULT NULL,' . + '`image_url` text COLLATE utf8_unicode_ci DEFAULT NULL,' . + '`title` varchar(255) COLLATE utf8_unicode_ci DEFAULT NULL,' . + '`author` varchar(255) COLLATE utf8_unicode_ci DEFAULT NULL,' . + '`album` varchar(255) COLLATE utf8_unicode_ci DEFAULT NULL,' . + '`type` varchar(255) COLLATE utf8_unicode_ci DEFAULT NULL,' . + '`time` smallint(5) DEFAULT NULL,' . + 'PRIMARY KEY (`id`), KEY `sid` (`sid`))'; + $db_results = Dba::write($sql); + self::set_version('db_version', '360011'); + } + } // end update class ?> diff --git a/modules/flash/xspf_player.php b/modules/flash/xspf_player.php index 42d9bdf6..9f9dde2e 100644 --- a/modules/flash/xspf_player.php +++ b/modules/flash/xspf_player.php @@ -35,8 +35,9 @@ switch ($_REQUEST['action']) { // Set for hackage! $_REQUEST['flash_hack'] = 1; $objects = $GLOBALS['user']->playlist->get_items(); - $stream = new Stream('xspf',$objects); - $stream->start(); + $stream = new Stream_Playlist(); + $stream->add($objects); + $stream->generate_playlist('xspf', false); break; case 'show': $play_url = Config::get('web_path') . '/modules/flash/xspf_player.php'; diff --git a/play/index.php b/play/index.php index e7c3917c..fb50a6ce 100644 --- a/play/index.php +++ b/play/index.php @@ -39,10 +39,19 @@ ob_end_clean(); /* These parameters had better come in on the url. */ $uid = scrub_in($_REQUEST['uid']); -$oid = $_REQUEST['song'] ? scrub_in($_REQUEST['song']) : scrub_in($_REQUEST['oid']); +$oid = $_REQUEST['oid'] + // FIXME: Any place that doesn't use oid should be fixed + ? scrub_in($_REQUEST['oid']) + : scrub_in($_REQUEST['song']); $sid = scrub_in($_REQUEST['ssid']); $xml_rpc = scrub_in($_REQUEST['xml_rpc']); $video = make_bool($_REQUEST['video']); +$type = scrub_in($_REQUEST['type']); + +if ($type == 'playlist') { + $playlist_type = scrub_in($_REQUEST['playlist_type']); + $oid = $sid; +} /* This is specifically for tmp playlist requests */ $demo_id = scrub_in($_REQUEST['demo_id']); @@ -120,6 +129,18 @@ if (Config::get('access_control')) { } } // access_control is enabled +// Handle playlist downloads +if ($type == 'playlist') { + $playlist = new Stream_Playlist($oid); + // Some rudimentary security + if ($uid != $playlist->user) { + access_denied(); + exit; + } + $playlist->generate_playlist($playlist_type, false); + exit; +} + /** * If we've got a tmp playlist then get the * current song, and do any other crazyness @@ -136,26 +136,26 @@ switch ($_REQUEST['action']) { break; case 'democratic': // Don't let them loop it + // FIXME: This looks hacky if (Config::get('play_type') == 'democratic') { Config::set('play_type', 'stream', true); } default: - if (Config::get('play_type') == 'stream') { + $stream_type = Config::get('play_type'); + if ($stream_type == 'stream') { $stream_type = Config::get('playlist_type'); } - else { - $stream_type = Config::get('play_type'); - } break; } -/* Start the Stream */ debug_event('stream.php' , 'Stream Type: ' . $stream_type . ' Media IDs: '. json_encode($media_ids), 5); -$stream = new Stream($stream_type, $media_ids); +$playlist = new Stream_Playlist(); +$playlist->add($media_ids); if (isset($urls)) { - $stream->add_urls($urls); + $playlist->add_urls($urls); } -$stream->start(); +// Depending on the stream type, will either generate a redirect or actually do +// the streaming. +$playlist->generate_playlist($stream_type, true); -} // end method switch ?> |